Java套接字服务器
我有一个Java进程正在使用java.io.ServerSocket在TCP套接字上创建一个侦听器,如下所示(简化):
ServerSocket server = new ServerSocket(4444,20);
server.accept();
Java Process在收到请求时触发工作线程,然后Worker使用java.io.PrintWriter和java.net.Socket发送JSON字符串:
PrintWriter out = new PrintWriter(clientSocket.getOutputStream());
out.println("JSON STRING");
out.flush();
out.close();
clientSocket.close();
我简化了Java代码,但这基本上就是它所做的一切。
.NET套接字客户端
接下来,我有一个.NET应用程序与托管此Java进程的计算机进行通信:
//Create Connection
Socket clientSocket = new Socket(AddressFamily.InterNetwork,
StreamType.Stream, ProtocolType.Tcp);
mySocket.Connect("192.168.1.102", 4444);
//Initialize Byte Buffer Larger than Expected JSON String and Read Bytes
byte[] receivedData = new byte[524288];
int numberOfBytes = clientSocket.Receive(receivedData, SocketFlags.None);
clientSocket.Close();
//Populate a new byte array of the exact size received with the received data
byte[] formatedBytes = new byte[numberOfBytes];
for (int i=0; i< numberOfBytes; i++)
{
formatedBytes[i] = receivedData[i];
}
//Convert Byte Array to String & Output Results
Response.ClearContent();
Response.ContentType("text/plain");
Response.Write(new System.Text.ASCIIEncoding().GetString(receivedData));
我的问题是,无论出于何种原因,当我尝试通过套接字流发送稍大的JSON字符串时,此实现不起作用。对于小数据量(小于2KB),我已成功测试了此实现,超过100个客户端连接和接收数据没有任何问题,但是尝试将JSON字符串大小增加到大约256KB会导致.NET应用程序截断结果。增加字节缓冲区数组的大小也无济于事 - 似乎.NET应用程序在传输所有数据之前删除了连接,或者Java应用程序没有使用PrintWriter发送整个String。
对此问题的任何见解将不胜感激 - 如果我自己取得任何进展,我会发布任何更新。
这是我遇到的解决方案 - 服务器现在工作得很好!再次感谢!
byte[] receivedData = new byte[512000]; // 4 Meg Buffer
Socket mySocket = new Socket(AddressFamily.InterNetwork,
SocketType.Stream, ProtocolType.Tcp);
mySocket.Connect("172.26.190.205", 4444);
mySocket.ReceiveBufferSize = 8192;
int numberOfBytesRead = 0;
int totalNumberOfBytes = 0;
do
{
numberOfBytesRead = mySocket.Receive(receivedData,totalNumberOfBytes ,
mySocket.ReceiveBufferSize,SocketFlags.None);
totalNumberOfBytes += numberOfBytesRead;
}
while (numberOfBytesRead > 0);
mySocket.Close();
byte[] formatedBytes = new byte[totalNumberOfBytes ];
for (int i = 0; i < totalNumberOfBytes ; i++)
{
formatedBytes[i] = receivedData[i];
}
Response.ClearContent();
Response.ContentType = "text/plain";
Response.Write(new System.Text.ASCIIEncoding().GetString(formatedBytes));
答案 0 :(得分:3)
虽然TCP在概念上是流,但底层IP发送和接收数据包。对于应用程序,这意味着始终从循环中的读取,因为发送方单个“写入”可能会导致接收方多次“读取”,并且另一种方式。
这归结为这样一个事实,即由TCP / IP堆栈实现如何将您发送的数据拆分成数据包。接收方,也就是网络堆栈,不知道预期有多少字节。它只需要它获取的内容并唤醒等待该套接字的进程,如果有的话,或者以其他方式缓冲字节。
您的情况很简单 - 只需继续读取套接字,直到看到EOF
,即读取零字节。但一般而言,您需要一个应用程序级协议,该协议可定义消息长度,为消息边界建立规则,或将显式长度信息嵌入到消息本身中。
答案 1 :(得分:0)
//初始化字节缓冲区大于 预期的JSON字符串和读取字节 byte [] receivedData = new 字节[524288]; int numberOfBytes = clientSocket.Receive(receivedData, SocketFlags.None); clientSocket.Close();
好的,它读取到达缓冲区的所有字节。然后关闭了流。窗口。
阅读文档 - 了解Receive方法。你必须反复调用它,直到你得到更多的数据。它会阻塞,直到有些数据到达,然后返回。它不会等到所有数据都到达。