Java的InputStream破坏信息,为什么?
我正在构建的服务器出现问题。我用Java将问题简化为一个简单的服务器-客户机对。我正在发送字节0到255。问题是InputStream将128 0x80到159 0x9F之间的值作为63 0x3F接收。这会毁了我的二进制传输。为什么会发生这种情况 这是我得到的输出,看看值127之后有多少个63:Java的InputStream破坏信息,为什么?,java,sockets,inputstream,Java,Sockets,Inputstream,我正在构建的服务器出现问题。我用Java将问题简化为一个简单的服务器-客户机对。我正在发送字节0到255。问题是InputStream将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
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
这是我的服务器代码,基于KnockServer示例:
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();
}
}
运行这个,如果你遇到同样的奇怪问题,告诉我。请帮帮我。谢谢大家! 来自PrintWriter的文档 将对象的格式化表示打印到文本输出流。 此类实现PrintStream中的所有打印方法。 它不包含写入原始字节的方法,因为程序 应使用未编码的字节流
因此,您可能使用了错误的工具来完成此工作。来自PrintWriter的文档 将对象的格式化表示打印到文本输出流。 此类实现PrintStream中的所有打印方法。 它不包含写入原始字节的方法,因为程序 应使用未编码的字节流
因此,您可能对此作业使用了错误的工具。您正在读取字节并发送字符。在Java中,它们的大小不同。字符是2个字节,而字节是单个的。如果只想发送字节,则必须使用OutputStream,而不是PrintWriter
还要记住,字节是从-128到128进行签名的,所以不能期望128以上的数字是正数。您必须反转签名数学。您正在读取字节并发送字符。在Java中,它们的大小不同。字符是2个字节,而字节是单个的。如果只想发送字节,则必须使用OutputStream,而不是PrintWriter
还要记住,字节是从-128到128进行签名的,所以不能期望128以上的数字是正数。您必须反转签名数学。不要使用字符发送字节。java不是C++,所以char不是8bit类型。一旦基本ASCII集合127用完,就开始发送Unicode内容 删除PrintWriter并在客户端中使用下一个循环:
for (int i = 0; i < 256; i++) {
echoSocket.getOutputStream().write(i);
}
不要使用字符发送字节。java不是C++,所以char不是8bit类型。一旦基本ASCII集合127用完,就开始发送Unicode内容 删除PrintWriter并在客户端中使用下一个循环:
for (int i = 0; i < 256; i++) {
echoSocket.getOutputStream().write(i);
}
您看到63s的原因是您的PrintWriter正在尝试从Unicode转换为ISO-8859-1,但字符128到159不是有效的Unicode字符,因此它们被转换为问号字符。您看到63s的原因是您的PrintWriter正在尝试从Unicode转换为ISO-8859-1,但字符128到159不是有效的Unicode字符,因此它们被转换为问号字符。+1而且他在另一端也使用了一个BufferedReader,这将有同样的问题。在处理二进制数据时,不要使用任何Reader或Writer类~即使使用PrintWriter,另一端也应该使用匹配的读取器。现在的代码写入字符,并接收字节@欧内斯特·弗里德曼·希尔,他没有在另一端使用BufferedReader。@不,你想再次检查客户端代码吗?我明白你在想什么-我不是说同样的管道,只是客户犯了同样的错误。服务器使用的是原始输出流,客户端使用的是BufferedReader。我指的是服务器,另一端。所以我们可能都是对的:-+1,他在另一端也使用了一个BufferedReader,这将有同样的问题。在处理二进制数据时,不要使用任何Reader或Writer类~即使使用PrintWriter,另一端也应该使用匹配的读取器。现在的代码写入字符,并接收字节@欧内斯特·弗里德曼·希尔,他没有在另一端使用BufferedReader。@不,你想再次检查客户端代码吗?我明白你在想什么-我不是说同样的管道,只是客户犯了同样的错误。服务器使用的是原始输出流,客户端使用的是BufferedReader。我指的是服务器,另一端。所以我们可能都是对的:-