2012-02-25 57 views
1

我遇到了我正在构建的服务器的问题。我将这个问题简化为一个简单的Java服务器 - 客户端对。我发送字节0到255.问题是从InputStream将128(0x80)到159(0x9F)的值作为63(0x3F)接收。这是杀死我的二进制转移。为什么会发生这种情况?Java的InputStream破坏信息,为什么?

此输出我得到的,怎么看值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

从为PrintWriter的文件在对象的文本输出流http://docs.oracle.com/javase/1.4.2/docs/api/java/io/PrintWriter.html

打印格式表示。 该类实现了PrintStream中的所有打印方法。 它不包含写入原始字节的方法,程序 应使用未编码的字节流。

因此,您可能使用了错误的工具来完成这项工作。

+0

+1,而且他在另一端也使用了'BufferedReader',这也会有同样的问题。在使用二进制数据时,不要使用任何'Reader'或'Writer'类〜 – 2012-02-25 21:59:43

+0

即使使用PrintWriter,另一端也应该使用匹配的Reader。现在的代码写入char,并接收字节。 @ ErnestFriedman-Hill,他没有在另一端使用BufferedReader。 – nos 2012-02-25 22:03:39

+0

@mvds嘿谢谢! – marcoantap 2012-02-25 22:29:40

0

您正在读取字节,并发送字符。在Java中,它们的大小不一样。字符是2个字节,并且它们是单个字节。如果你想发送简单的字节,你必须使用OutputStream,而不是PrintWriter。

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

1

不要使用字符发送字节。 Java不是C++,所以char不是8bit类型。一旦你用完基本的ASCII设置(127),你开始发送Unicode的东西。

下降PrintWriter,并在客户端改用下一个循环:

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

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