我正在构建一个服务器的问题。我将问题简化为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();
}
}
运行此命令告诉我你是否遇到了同样奇怪的问题。请帮我。谢谢!
答案 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以上的数字作为正数。你必须逆转签名的数学。