Java NIO学习与记录(四): SocketChannel与BIO服务器
SocketChannel可以创建连接TCP服务的客户端,用于为服务发送数据,SocketChannel的写操作和连接操作在非阻塞模式下不会发生阻塞,这篇文章里的客户端采用SocketChannel实现,利用线程池模拟多个客户端并发访问服务端的情景。服务端仍然采用ServerSocket来实现,主要用来看下阻塞模式下的服务端在并发访问时所做出的的处理。
一、使用SocketChannel实现一个客户端
private static ExecutorService ctp = Executors.newCachedThreadPool();
public static void main(String[] args) {
for (int i = 0; i < 10; i++) {
ctp.submit(IOTest::client); //并发十个客户端连接过去
}
}
public static void client() {
ByteBuffer buffer = ByteBuffer.allocate(1024); //定义缓冲区
SocketChannel socketChannel = null;
try {
socketChannel = SocketChannel.open(); //打开SocketChannel
socketChannel.configureBlocking(false); //设置为非阻塞模式
socketChannel.connect(new InetSocketAddress("127.0.0.1", 2333)); //连接服务
while (true) {
if(socketChannel.finishConnect()){ //这里的finishConnect是尝试连接,有可能返回false,因此使用死循环进行连接检查,确保连接已经正常建立。
System.out.println("客户端已连接到服务器");
int i = 0;
while (i < 5) {
TimeUnit.SECONDS.sleep(1); //隔一秒钟写一条
String info = "来自客户端的第" + (i++) + "条消息";
buffer.clear();
buffer.put(info.getBytes());
buffer.flip();
while (buffer.hasRemaining()) {
socketChannel.write(buffer); //给服务写消息
}
}
break;
}
}
} catch (IOException | InterruptedException e) {
e.printStackTrace();
} finally {
try {
if (socketChannel != null) {
System.out.println("客户端Channel关闭");
socketChannel.close();
}
} catch (IOException e) {
e.printStackTrace();
}
}
上面会同时产生10个客户端去连接服务端
二、使用ServerSocket实现一个BIO的TCP服务
ServerSocket serverSocket = null;
int recvMsgSize = 0;
InputStream in = null;
try {
serverSocket = new ServerSocket(2333); //开一个监听2333端口的TCP服务
byte[] recvBuf = new byte[1024];
while (true) {
Socket clntSocket = serverSocket.accept(); //探听有没有新的客户端连接进来,没有就阻塞
SocketAddress clientAddress = clntSocket.getRemoteSocketAddress(); //通过跟服务连接上的客户端socket,拿到客户端地址
System.out.println("连接成功,处理客户端:" + clientAddress);
in = clntSocket.getInputStream(); //数据流
while ((recvMsgSize = in.read(recvBuf)) != -1) { //读取发送的数据,当客户端未断开连接,且不往服务端发数据的时候,说明一直处于准备读的状态,会一直阻塞下去,直到有数据写入(读就绪)
byte[] temp = new byte[recvMsgSize];
System.arraycopy(recvBuf, 0, temp, 0, recvMsgSize);
System.out.println("收到客户端" + clientAddress + "的消息内容:" + new String(temp)); //打印消息
}
System.out.println("-----------------------------------");
}
} catch (IOException e) {
e.printStackTrace();
} finally {
try {
if (serverSocket != null) {
System.out.println("socket关闭!");
serverSocket.close();
}
if (in != null) {
System.out.println("stream连接关闭!");
in.close();
}
} catch (IOException e) {
e.printStackTrace();
}
}
运行上面的代码,服务端打印如下:
连接成功,处理客户端:/127.0.0.1:54688
收到客户端/127.0.0.1:54688的消息内容:来自客户端的第0条消息
收到客户端/127.0.0.1:54688的消息内容:来自客户端的第1条消息
收到客户端/127.0.0.1:54688的消息内容:来自客户端的第2条消息
收到客户端/127.0.0.1:54688的消息内容:来自客户端的第3条消息
收到客户端/127.0.0.1:54688的消息内容:来自客户端的第4条消息
-----------------------------------
连接成功,处理客户端:/127.0.0.1:54680
收到客户端/127.0.0.1:54680的消息内容:来自客户端的第0条消息来自客户端的第1条消息来自客户端的第2条消息来自客户端的第3条消息来自客户端的第4条消息
-----------------------------------
连接成功,处理客户端:/127.0.0.1:54689
收到客户端/127.0.0.1:54689的消息内容:来自客户端的第0条消息来自客户端的第1条消息来自客户端的第2条消息来自客户端的第3条消息来自客户端的第4条消息
-----------------------------------
连接成功,处理客户端:/127.0.0.1:54682
收到客户端/127.0.0.1:54682的消息内容:来自客户端的第0条消息来自客户端的第1条消息来自客户端的第2条消息来自客户端的第3条消息来自客户端的第4条消息
-----------------------------------
连接成功,处理客户端:/127.0.0.1:54683
收到客户端/127.0.0.1:54683的消息内容:来自客户端的第0条消息来自客户端的第1条消息来自客户端的第2条消息来自客户端的第3条消息来自客户端的第4条消息
-----------------------------------
连接成功,处理客户端:/127.0.0.1:54684
收到客户端/127.0.0.1:54684的消息内容:来自客户端的第0条消息来自客户端的第1条消息来自客户端的第2条消息来自客户端的第3条消息来自客户端的第4条消息
-----------------------------------
连接成功,处理客户端:/127.0.0.1:54685
收到客户端/127.0.0.1:54685的消息内容:来自客户端的第0条消息来自客户端的第1条消息来自客户端的第2条消息来自客户端的第3条消息来自客户端的第4条消息
-----------------------------------
连接成功,处理客户端:/127.0.0.1:54681
收到客户端/127.0.0.1:54681的消息内容:来自客户端的第0条消息来自客户端的第1条消息来自客户端的第2条消息来自客户端的第3条消息来自客户端的第4条消息
-----------------------------------
连接成功,处理客户端:/127.0.0.1:54686
收到客户端/127.0.0.1:54686的消息内容:来自客户端的第0条消息来自客户端的第1条消息来自客户端的第2条消息来自客户端的第3条消息来自客户端的第4条消息
-----------------------------------
连接成功,处理客户端:/127.0.0.1:54687
收到客户端/127.0.0.1:54687的消息内容:来自客户端的第0条消息来自客户端的第1条消息来自客户端的第2条消息来自客户端的第3条消息来自客户端的第4条消息
-----------------------------------
可以看到,消息是按照顺序,一个一个连接进来,然后完成处理的,至于后面的消息为什么会被合并成一个,也是这个原因,因为阻塞,所以等第一个连接逐条输出完成后,第二个连接进来,这时很可能客户端的SocketChannel已经将十条消息全部写入channel,等第一个连接处理完成后,接到第二条消息时就已经是全部的消息了,因此一次性输出,后面的合并也是这个原因(主要客户端使用NIO实现,因此写和连接服务不会发生阻塞,因此在第次个请求服务端还在处理时,其余的客户端数据也在执行并写入通道,最终服务端处理完第一个连接,然后继续接收第二个连接时,数据便是完整的5条数据了)。
上面的服务端是一个典型的阻塞IO的服务,accept在没有连接进来时会发生阻塞,read在客户端连接没关闭,且不再写消息时,服务端的read将一直处于读等待状态并阻塞,直到收到新的消息转为读就绪才会继续往下执行(这就是上面例子里第一个进来的连接可以逐条输出的原因),完全串行化,过程如下图:

图1
下面,来改造下服务端,让其处理能力更好一些,除了accept,下面的处理逻辑全部交给线程池处理:
while (true) {
Socket clntSocket = serverSocket.accept(); //探听有没有新的客户端连接进来,没有就阻塞
SocketAddress clientAddress = clntSocket.getRemoteSocketAddress(); //通过跟服务连接上的客户端socket,拿到客户端地址
System.out.println("连接成功,处理客户端:" + clientAddress);
ctp.execute(() -> {
int recvMsgSize = 0;
InputStream in = null; //数据流
try {
in = clntSocket.getInputStream();
while ((recvMsgSize = in.read(recvBuf)) != -1) { //读取发送的数据,当客户端未断开连接,且不往服务端发数据的时候,说明一直处于准备读的状态,会一直阻塞下去,直到有数据写入(读就绪)
byte[] temp = new byte[recvMsgSize];
System.arraycopy(recvBuf, 0, temp, 0, recvMsgSize);
System.out.println("收到客户端" + clientAddress + "的消息内容:" + new String(temp)); //打印消息
}
System.out.println("-----------------------------------");
} catch (IOException e) {
e.printStackTrace();
}
});
}
运行结果:
连接成功,处理客户端:/127.0.0.1:55259
连接成功,处理客户端:/127.0.0.1:55265
连接成功,处理客户端:/127.0.0.1:55266
连接成功,处理客户端:/127.0.0.1:55257
连接成功,处理客户端:/127.0.0.1:55260
连接成功,处理客户端:/127.0.0.1:55258
连接成功,处理客户端:/127.0.0.1:55261
连接成功,处理客户端:/127.0.0.1:55262
连接成功,处理客户端:/127.0.0.1:55263
连接成功,处理客户端:/127.0.0.1:55264
收到客户端/127.0.0.1:55265的消息内容:来自客户端的第0条消息
收到客户端/127.0.0.1:55266的消息内容:来自客户端的第0条消息
收到客户端/127.0.0.1:55258的消息内容:来自客户端的第0条消息
收到客户端/127.0.0.1:55257的消息内容:来自客户端的第0条消息
收到客户端/127.0.0.1:55261的消息内容:来自客户端的第0条消息
收到客户端/127.0.0.1:55262的消息内容:来自客户端的第0条消息
收到客户端/127.0.0.1:55263的消息内容:来自客户端的第0条消息
收到客户端/127.0.0.1:55260的消息内容:来自客户端的第0条消息
收到客户端/127.0.0.1:55259的消息内容:来自客户端的第0条消息
收到客户端/127.0.0.1:55264的消息内容:来自客户端的第0条消息
收到客户端/127.0.0.1:55265的消息内容:来自客户端的第1条消息
收到客户端/127.0.0.1:55266的消息内容:来自客户端的第1条消息
收到客户端/127.0.0.1:55258的消息内容:来自客户端的第1条消息
收到客户端/127.0.0.1:55257的消息内容:来自客户端的第1条消息
收到客户端/127.0.0.1:55261的消息内容:来自客户端的第1条消息
收到客户端/127.0.0.1:55260的消息内容:来自客户端的第1条消息
收到客户端/127.0.0.1:55262的消息内容:来自客户端的第1条消息
收到客户端/127.0.0.1:55263的消息内容:来自客户端的第1条消息
收到客户端/127.0.0.1:55259的消息内容:来自客户端的第1条消息
收到客户端/127.0.0.1:55264的消息内容:来自客户端的第1条消息
收到客户端/127.0.0.1:55266的消息内容:来自客户端的第2条消息
收到客户端/127.0.0.1:55262的消息内容:来自客户端的第2条消息
收到客户端/127.0.0.1:55261的消息内容:来自客户端的第2条消息
收到客户端/127.0.0.1:55260的消息内容:来自客户端的第2条消息
收到客户端/127.0.0.1:55263的消息内容:来自客户端的第2条消息
收到客户端/127.0.0.1:55257的消息内容:来自客户端的第2条消息
收到客户端/127.0.0.1:55265的消息内容:来自客户端的第2条消息
收到客户端/127.0.0.1:55258的消息内容:来自客户端的第2条消息
收到客户端/127.0.0.1:55259的消息内容:来自客户端的第2条消息
收到客户端/127.0.0.1:55264的消息内容:来自客户端的第2条消息
收到客户端/127.0.0.1:55258的消息内容:来自客户端的第3条消息
收到客户端/127.0.0.1:55266的消息内容:来自客户端的第3条消息
收到客户端/127.0.0.1:55257的消息内容:来自客户端的第3条消息
收到客户端/127.0.0.1:55262的消息内容:来自客户端的第3条消息
收到客户端/127.0.0.1:55261的消息内容:来自客户端的第3条消息
收到客户端/127.0.0.1:55263的消息内容:来自客户端的第3条消息
收到客户端/127.0.0.1:55265的消息内容:来自客户端的第3条消息
收到客户端/127.0.0.1:55260的消息内容:来自客户端的第3条消息
收到客户端/127.0.0.1:55264的消息内容:来自客户端的第3条消息
收到客户端/127.0.0.1:55259的消息内容:来自客户端的第3条消息
收到客户端/127.0.0.1:55266的消息内容:来自客户端的第4条消息
收到客户端/127.0.0.1:55265的消息内容:来自客户端的第4条消息
-----------------------------------
收到客户端/127.0.0.1:55263的消息内容:来自客户端的第4条消息
收到客户端/127.0.0.1:55261的消息内容:来自客户端的第4条消息
-----------------------------------
收到客户端/127.0.0.1:55260的消息内容:来自客户端的第4条消息
-----------------------------------
收到客户端/127.0.0.1:55262的消息内容:来自客户端的第4条消息
-----------------------------------
收到客户端/127.0.0.1:55257的消息内容:来自客户端的第4条消息
-----------------------------------
-----------------------------------
收到客户端/127.0.0.1:55258的消息内容:来自客户端的第4条消息
-----------------------------------
-----------------------------------
收到客户端/127.0.0.1:55264的消息内容:来自客户端的第4条消息
-----------------------------------
收到客户端/127.0.0.1:55259的消息内容:来自客户端的第4条消息
-----------------------------------
消息被分开了,接收连接虽然仍然是串行,但实际的处理速度在多线程的帮助下已经比之前快很多了,流程如下图:

图2
三、BIO总结
综合看下来,传统的阻塞IO,按照图2的方式进行,虽然利用多线程避免了read等操作的阻塞对accept的影响,提高了处理效率,但想象下,如果现在存在高并发的情况,图2的模型如果不使用线程池,就会创建大量线程,会发生大量的线程上下文切换,影响整体效率,并且会影响新的线程,如果使用线程池,虽然某种程度上避免了线程的创建和上下文切换的量级,但是在大量并发的场景下,会发生排队,一旦发生排队,紧接着就会影响到accept。
Java NIO学习与记录(四): SocketChannel与BIO服务器的更多相关文章
- Java NIO学习与记录(八): Reactor两种多线程模型的实现
Reactor两种多线程模型的实现 注:本篇文章例子基于上一篇进行:Java NIO学习与记录(七): Reactor单线程模型的实现 紧接着上篇Reactor单线程模型的例子来,假设Handler的 ...
- Java NIO 学习笔记(四)----文件通道和网络通道
目录: Java NIO 学习笔记(一)----概述,Channel/Buffer Java NIO 学习笔记(二)----聚集和分散,通道到通道 Java NIO 学习笔记(三)----Select ...
- Java NIO学习与记录(六): NIO线程模型
NIO线程模型 上一篇说的是基于操作系统的IO处理模型,那么这一篇来介绍下服务器端基于IO模型和自身线程的处理方式. 一.传统阻塞IO模型下的线程处理模式 这种处理模型是基于阻塞IO进行的,上一篇讲过 ...
- Java NIO学习与记录(五): 操作系统的I/O模型
操作系统的I/O模型 在开始介绍NIO Reactor模式之前,先来介绍下操作系统的五种I/O模型,了解了这些模型,对理解java nio会有不小的帮助. 先来看下一个服务端处理一次网络请求的流程图: ...
- Java NIO学习与记录(二):FileChannel与Buffer用法与说明
FileChannel与Buffer用法与说明 上一篇简单介绍了NIO,这一篇将介绍FileChannel结合Buffer的用法,主要介绍Buffer FileChannel的简单使用&Buf ...
- Java NIO学习与记录(一):初识NIO
初识 工作中有些地方用到了netty,netty是一个NIO框架,对于NIO却不是那么熟悉,这个系列的文章是我在学习NIO时的一个记录,也期待自己可以更好的掌握NIO. 一.NIO是什么? 非阻塞式I ...
- Java NIO学习与记录(七): Reactor单线程模型的实现
Reactor单线程模型的实现 一.Selector&Channel 写这个模型需要提前了解Selector以及Channel,之前记录过FileChannel,除此之外还有以下几种Chann ...
- Java NIO学习与记录(三): Scatter&Gather介绍及使用
Scatter&Gather介绍及使用 上一篇知道了Buffer的工作机制,以及FileChannel的简单用法,这一篇介绍下 Scatter&Gather 1.Scatter(分散 ...
- Java NIO 学习笔记(三)----Selector
目录: Java NIO 学习笔记(一)----概述,Channel/Buffer Java NIO 学习笔记(二)----聚集和分散,通道到通道 Java NIO 学习笔记(三)----Select ...
随机推荐
- python 探测网站目录的GUI程序-乾颐堂
1.pyqt4写的界面 find_ui.py 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 ...
- marioTcp
https://github.com/nicholaszj/marioTcp MarioTCP MarioTCP 是使用libevent模型来建立的一个性能强大的TCP服务器. 1:Getting S ...
- SpringMvc 全局异常处理器定义,友好的返回后端错误信息
import com.google.common.collect.Maps; import org.apache.log4j.Logger; import org.springframework.be ...
- Android实现求和运算
实验要求: 用Android实现一个界面,在该页面点击实现加法运算. 代码实现 实现结果 输入结果为空时,如图 问题及解决 函数中使用了强制转换,当输入字符串是也能转换为int型数据,但是当输入字符时 ...
- Vivado&ISE&Quartus II调用Modelsim级联仿真
博主一直致力寻找高效的工作方式,所以一直喜欢折腾软件,从刚开始只用软件IDE自带的编辑器,到Notepad++,再到后来的Vim,从用ISE14.7自带的Isim仿真,到发现更好的Modelsim,再 ...
- EJB3.0 EJB开发消息驱动bean
(7)EJB3.0 EJB开发消息驱动bean JMS 一: Java消息服务(Java Message Service) 二:jms中的消息 消息传递系统的中心就是消息.一条 Message 由三个 ...
- Linux Guard Service - 进程分裂与脱离
进程分裂更名 void set_ps_name(char *name) { prctl(PR_SET_NAME, name); } 修改进程长名称 备份进程环境变量空间 for (i = 1; i & ...
- Android studio 报错 installation failed with message failed to finalize session:INSTALL_FAILED_INVALID_APK 解决方法
解决方案: File->Setting->Build->Instant Run
- js如何给当前日期+1?
一天=24小时=1440分钟=86400秒 所以给当前日期加一天的步骤为: 1.获取当前日期: 2.利用86400秒给其进行加一天操作: 3.类似加一天,两天,一月,一年等,过程如此. 代码如下(以j ...
- 自己从0开始学习Unity的笔记 VIII (C#中类继承练习 II)
自己写了一个关于兵种的,因为一直在测试,到底面向对象是个什么玩意...然后就做了这个 namespace 兵种 { class Role //作为父类,构建一个普通角色属性用于继承 { protect ...