Java InputStream会破坏信息,为什么?

时间:2012-02-25 21:53:22

标签: java sockets inputstream

我正在构建一个服务器的问题。我将问题简化为Java中的简单服务器 - 客户端对。我发送的字节数为0到255.问题是输入流数据接收的值为128(0x80)到159(0x9F)为63(0x3F)。这会杀死我的二进制转移。为什么会发生这种情况?

这是我得到的输出,看看值127之后是如何有一堆63的:

0 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29
30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56
 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 8
3 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 10
7 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 12
7 63 63 63 63 63 63 63 63 63 63 63 63 63 63 63 63 63 63 63 63 63 63 63 63 63 63
63 63 63 63 63 63 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 17
5 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 19
5 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 21
5 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 23
5 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 25
5

这是我的服务器代码,基于KnockKnockServer示例:

import java.net.*;
import java.io.*;

public class Server {
        public static void main(String[] args) throws IOException {

                ServerSocket serverSocket = null;
                try {
                        serverSocket = new ServerSocket(4444);
                } catch (IOException e) {
                        System.err.println("Could not listen on port: 4444.");
                        System.exit(1);
                }

                Socket clientSocket = null;
                try {
                        clientSocket = serverSocket.accept();
                } catch (IOException e) {
                        System.err.println("Accept failed.");
                        System.exit(1);
                }

                OutputStream out = clientSocket.getOutputStream();
                InputStream in = clientSocket.getInputStream();

                while (true)
                {
                    int val = in.read();
                    if (val < 0)
                        continue;

                    System.out.print(val);
                    System.out.print(' ');
                    if (val == 255)
                        break;
                }

                out.close();
                in.close();
                clientSocket.close();
                serverSocket.close();
        }
}

这是我的客户端代码,基于EchoClient示例:

import java.io.*;
import java.net.*;

public class Client {
        public static void main(String[] args) throws IOException {

                Socket echoSocket = null;
                PrintWriter out = null;
                BufferedReader in = null;

                try {
                        echoSocket = new Socket("localhost", 4444);
                        out = new PrintWriter(echoSocket.getOutputStream(), true);
                        in = new BufferedReader(new InputStreamReader(
                                                                                echoSocket.getInputStream()));
                } catch (UnknownHostException e) {
                        System.err.println("Don't know about host: taranis.");
                        System.exit(1);
                } catch (IOException e) {
                        System.err.println("Couldn't get I/O for "
                                                             + "the connection to: taranis.");
                        System.exit(1);
                }

        for (int i = 0; i < 256; i++)
            out.print((char) i);
        out.flush();

        out.close();
        in.close();
        echoSocket.close();
        }
}

运行此命令告诉我你是否遇到了同样奇怪的问题。请帮我。谢谢!

4 个答案:

答案 0 :(得分:4)

来自http://docs.oracle.com/javase/1.4.2/docs/api/java/io/PrintWriter.html

的PrintWriter文档
  

将对象的格式化表示打印到文本输出流。   此类实现PrintStream中的所有打印方法。   它不包含用于编写程序的原始字节的方法   应该使用未编码的字节流。

所以你可能正在使用错误的工具来完成这项工作。

答案 1 :(得分:1)

不要使用char来发送字节。 Java不是C ++,因此char不是8位类型。一旦用完基本的ASCII集(127),就开始发送Unicode内容。

删除PrintWriter并在您的客户端中使用下一个循环:

    for (int i = 0; i < 256; i++) {
        echoSocket.getOutputStream().write(i);
    }

答案 2 :(得分:1)

您看到63s的原因是您的PrintWriter正在尝试从Unicode转换为ISO-8859-1,但字符128到159不是有效的Unicode字符,因此它们会转换为问号字符。

答案 3 :(得分:0)

您正在读取字节并发送字符。在Java中,它们的大小不同。字符是2个字节,字节很好,它们是单个字节。如果你想简单地发送字节,你将不得不使用OutputStream,而不是PrintWriter。

还记得字节是从-128到128签名的,所以你不能期望得到128以上的数字作为正数。你必须逆转签名的数学。