直接干代码,用BIO写一个Server端,然后使用telnet模拟客户端发送数据
import java.io.IOException;
import java.io.InputStream;
import java.net.ServerSocket;
import java.net.Socket;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;public class BioServer {
public static void main(String[] args) throws IOException {
// 1.创建一个线程池
ExecutorService executorServices = Executors.newCachedThreadPool();
// 2. 创建一个ServerSocket服务端
ServerSocket serverSocket = new ServerSocket(8888);
System.out.println("服务端启动成功!"); while (true) {
System.out.println("线程:" + Thread.currentThread().getId() + "-等待客服端连接。。。");
final Socket socket = serverSocket.accept(); // 如果没有客户端与server端建立连接,这里会一直阻塞
System.out.println("连接到一个客户端");
executorServices.submit(() -> bzHandler(socket));
}
} /**
* 业务方法,与客户端通信
*
* @param socket
*/
public static void bzHandler(Socket socket) {
try {
byte[] bytes = new byte[1024];
// 通过socket获取输入流
InputStream inputStream = socket.getInputStream(); // 循环读取输入流中的数据
while (true) {
System.out.println("线程:" + Thread.currentThread().getId() + "-等着读取客户端输入流中的内容。。。。");
int read = inputStream.read(bytes); //如果已经建立连接的客户端没有发送数据,这里会一直阻塞
if (read != -1) {
System.out.println(new String(bytes, 0, read));
} else {
break;
}
}
} catch (Exception e) {
e.printStackTrace();
} finally {
try {
socket.close();
} catch (IOException e) {
e.printStackTrace();
}
}
}
}
(1)程序启动,控制台打印如下:
服务端启动成功!
线程:1-等待客服端连接。。。
说明主线程阻塞在accept()这句代码这儿了。。。
(2)使用telnet建立连接后(注意还没有send数据到服务端),控制台打印如下:
服务端启动成功!
线程:1-等待客服端连接。。。
连接到一个客户端
线程:1-等待客服端连接。。。
线程:11-等着读取客户端输入流中的内容。。。。
连接到一个客户端 : 很容易理解
线程:11-等着读取客户端输入流中的内容。。。。: 说明创建了一个线程去处理与客户端的连接,同时程序也卡在了 inputStream.read(bytes)这儿,阻塞了!
线程:1-等待客服端连接。。。: 说明主线程accept一个socket连接之后,在交给子线程处理后,主线程又阻塞在了accept()方法这儿。。。(3)客户端发送数据到server端,控制台打印如下:
服务端启动成功!
线程:1-等待客服端连接。。。
连接到一个客户端
线程:1-等待客服端连接。。。
线程:11-等着读取客户端输入流中的内容。。。。
zhengqinfeng
线程:11-等着读取客户端输入流中的内容。。。。
前面的打印就不说了,
zhengqinfeng :说明server端接收到了client端发送过来的数据
线程:11-等着读取客户端输入流中的内容。。。。 : 说明子线程又阻塞到了read()方法处。。
(4) 再启一个client端 , 控制台打印如下:
服务端启动成功!
线程:1-等待客服端连接。。。
连接到一个客户端
线程:1-等待客服端连接。。。
线程:11-等着读取客户端输入流中的内容。。。。
zhengqinfeng
线程:11-等着读取客户端输入流中的内容。。。。
连接到一个客户端
3线程:1-等待客服端连接。。。
线程:12-等着读取客户端输入流中的内容。。。。
又是一个线程去处理新的连接 。。。
总结:
(1)BIO, 同步阻塞,主线程会阻塞,子线程同样也会阻塞。
(2)每一个客户端连接过来时,server端都会创建一个子线程去与之交互
补充: telnet发送请求
(1)telnet 127.0.0.1 8888
(2) ctrl+]
(3) send data