问题描述:使用socket编写一个简单的发送接收信息的demo,客户端往服务端发送信息但是服务端无法接收到,只有客户端断开socket服务端才能接收到之前客户端发送的信息。
场景:客户端使用tcp模拟器模拟发送信息,客户端希望能够实时接收到客户端发送的信息。
客户端:
服务端代码
import java.io.IOException;
import java.io.InputStream;
import java.io.OutputStream;
import java.net.ServerSocket;
import java.net.Socket;
public class SocketServer {
public static void main(String[] args) throws IOException{
int port = 666;
ServerSocket server = new ServerSocket(port);
while(true){
Socket sockect = server.accept();//保持阻塞状态,直到有连接进来
InputStream input = sockect.getInputStream();
int len;
byte[] bytes = new byte[1024];
StringBuilder string = new StringBuilder();
while((len = input.read(bytes))!=-1){
string.append(new String(bytes, 0, len, "UTF-8"));
}
System.out.println("收到客户端信息: " + string);
input.close();
output.close();
sockect.close();
}
}
}
实测,在客户端没有断开连接的情况下服务器无法收到消息。
原因分析:在input.read处阻塞。
解决方式
import java.io.IOException;
import java.io.InputStream;
import java.net.ServerSocket;
import java.net.Socket;
public class IOServer {
public static void main(String[] args) throws Exception {
ServerSocket serverSocket = new ServerSocket(8080);
// (1) 接收新连接线程
new Thread(() -> {
while (true) {
try {
// (1) 阻塞方法获取新的连接
Socket socket = serverSocket.accept();
// (2) 每一个新的连接都创建一个线程,负责读取数据
// new Thread(() -> {
try {
byte[] data = new byte[1024];
InputStream inputStream = socket.getInputStream();
while (true) {
int len;
// (3) 按字节流方式读取数据
while ((len = inputStream.read(data)) != -1) {
System.out.println(new String(data, 0, len));
}
}
} catch (IOException e) {
}
// }).start();
} catch (IOException e) {
}
}
}).start();
}
}
按照以上逻辑可以保持实时接收。