BIO
流程:
- 服务端建立seversocket并绑定端口号
- seversocket.accept()监听客户端
- 客户端建立socket并绑定访问IP以及端口号
- seversocket.accept()监听到客户端链接请求建立socket进行通信
- 双方通过OutputStream发送数据流,通过IntputStream接受数据流
- 最终释放资源。
服务端:
ServerSocket servers = new ServerSocket();
servers.bind(new InetSocketAddress(PORT));
Socket socket = servers.accept();
客户端:
Socket socket=new Socket();
socket.connect(new InetSocketAddress("127.0.0.1",9999));
发送文件:
OutputStream out=socket.getOutputStream();
File file=new File("AAA.txt");
InputStream in=new FileInputStream(file);//将文件读入内存中
byte[] fileBytes=new byte[100];
//循环分次发送
int len=-1;
while(len=in.read(fileBytes))!=-1){
out.write(fileBytes,0,len);
}
接受
OutputStream filesout=new FileOutputStream("BBB.txt");
byte[] fileBytes=new byte[100];
InputStream in=socket.getInputStream();
int len=-1;
while((len=in.read(fileBytes))!=-1){
filesout.write(fileBytes,0,len);
}
关闭
InputStream in=socket.getInputStream();
in.close();
OutputStream out=socket.getOutputStream();
out.close();
socket.close();
BIO将会在accept处造成阻塞
因此可以采用多线程循环方式来进行并发处理(在rpc部分已经有做出例子就懒得再弄了)
NIO(施工中)
- 打开ServerSocketChannel,监听客户端连接
- 绑定监听端口,设置连接为非阻塞模式
- 创建Reactor线程,创建多路复用器并启动线程
- 将ServerSocketChannel注册到Reactor线程中的Selector上,监听ACCEPT事件
- Selector轮询准备就绪的key
- Selector监听到新的客户端接入,处理新的接入请求,完成TCP三次握手,简历物理链路
- 设置客户端链路为非阻塞模式
- 将新接入的客户端连接注册到Reactor线程的Selector上,监听读操作,读取客户端发送的网络消息
- 异步读取客户端消息到缓冲区
- 对Buffer编解码,处理半包消息,将解码成功的消息封装成Task
- 将应答消息编码为Buffer,调用SocketChannel的write将消息异步发送给客户端
服务器端:
ServerSocketChannel ssc= ServerSocketChannel.open();//新建NIO通道
ssc.configureBlocking( false );//使通道为非阻塞
ServerSocket ss = ssc.socket();//创建基于NIO通道的socket连接
ss.bind(new InetSocketAddress("127.0.0.1",SERVERPORT));//新建socket通道的端口
//将NIO通道选绑定到择器,当然绑定后分配的主键为skey
SelectionKey skey = ssc.register( selector, SelectionKey.OP_ACCEPT );
- ssc.register( selector, SelectionKey.OP_ACCEPT );
这个方法是把ssc注册绑定到选择器selector 这样下次你想找ssc或者判断一个对象是不是ssc就可以通过selector来查找,查找是通过判断ssc的key得到的。
至于第二个参数SelectionKey.OP_ACCEPT 你可以理解成ssc的key类型或者操作权限
客户端: