socket的NIO操作
一、前言
Java中直接使用socket进行通信的场景应该不是很多,在公司的一个项目中有这种需求,所以根据自己的理解和相关资料的参考,基于NIO 实现了一组工具类库,具体的协议还未定义,后续再整理
二、实现思路
包结构如下:
Listener: 事件监听接口
AcceptListener(请求事件接口),TCPServerProtocol实现类中当服务端接收到连接请求并成功建立通信之后通知注册的此事件集合;
ReadListener(读取事件接口),TCPProtocol实现类中读取接收到的信息完成之后通知注册的此事件集合;
SendListener(发送事件接口),TCPProtocol实现类中调用发送信息方法之后通知注册的此事件集合;
Protocol:TCP处理接口
TCPProtocol(读取、输出TCP处理接口),定义了NIO中关于输入、输出处理以及相关监听事件的维护
TCPServerProtocol(接收TCP请求处理接口),定义了NIO中TCP请求处理以及相关监听事件的维护
Util:辅助类
HelperUtil(一些基本操作工具类),定义获取KEY值,获取本地IP,核对结束帧等
SocketConfig(socket连接配置类),定义IP地址、端口、处理接口、是否自动重置等配置信息
SocketStat(socket连接状态管理类),管理socket的生命周期,提供socket控制方法
SocketLogicException(socket异常类),定义此类库中可能出现的异常
Socket:对外服务类
NSocketBlockClient(socket阻塞客户端类),提供阻塞的客户端实现
NSocketClient(socket非阻塞客户端类),提供非阻塞的客户端实现
NSocketService(socket非阻塞服务端类),提供非阻塞的服务端实现
SelectOptionListener(发送事件接口的实现),用于改变selector的interestOps为OP_WRITE
SocketClientTask(客户端线程任务),用于管理客户端的输入、输出事件以及异常处理
SocketServiceTask(服务端线程任务),用于管理服务端的输入、输出、请求事件以及异常处理
核心类图:
三、使用方式
(1) 通过实现AcceptListener、ReadListener、SendListener事件接口来注入发送、读取、接收业务
(2) 通过设置SocketConfig对象属性来进行SOCKET通信配置
(3) 通过NSocketBlockClient、NSocketClient、NSocketService对象来进行信息发送
服务端例子:
SocketConfig config=new SocketConfig(true, "10.33.6.178", 8899);
TCPServerProtocol protocol=new DefaultServerProtocol(new AcceptListener(){
@Override
public void handleEvent(SocketStat socket) {
socket.getConfig().getProtocol().addReadListener(new TestRead());
} });
NSocketService server=new NSocketService(config, protocol);
其中 TestRead 为实现ReadListener的类
public void handleEvent(byte[] message,TCPProtocol tcpProtocol) {
System.out.println(new String(message,Charset.forName("GBK")));
tcpProtocol.sendMessage("hello".getBytes(Charset.forName("GBK")));
}
非阻塞客户端例子:
SocketConfig config=new SocketConfig(true, "10.33.6.178", 8899);
config.getProtocol().addReadListener(new ReadListener() { @Override
public void handleEvent(byte[] message, TCPProtocol tcpProtocol) { System.out.println("recive from server:"+ new String(message));
}
});
NSocketClient client=new NSocketClient(config); client.sendMessage("test nsocket".getBytes());
关于NIO处理的核心类:
public class DefaultServerProtocol implements TCPServerProtocol{
private LinkedList<AcceptListener> acceptList=new LinkedList<AcceptListener>();
public DefaultServerProtocol(AcceptListener... acceptColl){
for(AcceptListener al:acceptColl){
this.addAcceptListener(al);
}
}
@Override
public SocketStat handleAccept(SelectionKey key) throws IOException {
SocketChannel channel=((ServerSocketChannel)key.channel()).accept();
channel.configureBlocking(false);
Socket socket=channel.socket();
SocketConfig config=new SocketConfig(false,socket.getInetAddress().getHostAddress(),socket.getPort(),this.createProtocol());
SocketStat stat=new SocketStat(config, key.selector(), channel);
System.out.println("远程客户端地址:".concat(socket.getInetAddress().getHostAddress()));
notifyAccept(stat);
return stat;
}
@Override
public void addAcceptListener(AcceptListener al) {
this.acceptList.add(al);
}
@Override
public void notifyAccept(SocketStat socket) {
for(AcceptListener al: acceptList){
al.handleEvent(socket);
}
}
/**
*
* 获取关于TCP 的读取和写入操作协议,可以override 返回自己的实现
*
* @return
* @since Ver 1.0
*/
public TCPProtocol createProtocol(){
return new DefaultTCPProtocol();
}
}
public class DefaultTCPProtocol implements TCPProtocol {
/**读取缓存区*/
private ByteBuffer readBuff;
/**待发送消息队列*/
protected Queue<ByteBuffer> messageQueue=new LinkedBlockingQueue<ByteBuffer>();
/**锁*/
private Object lockObje=new Object();
/**读取监听*/
private LinkedList<ReadListener> readList=new LinkedList<ReadListener>();
/**发送监听*/
private LinkedList<SendListener> sendList=new LinkedList<SendListener>();
public DefaultTCPProtocol(ReadListener... readColl){
readBuff=ByteBuffer.allocate(1024);
for(ReadListener rl : readColl){
this.addReadListener(rl);
}
}
@Override
public void handleRead(SelectionKey key) throws IOException {
SocketChannel clientChn=(SocketChannel)key.channel();
ByteArrayOutputStream out=new ByteArrayOutputStream();
try{
synchronized(this.readBuff){
readBuff.clear();
int bytesRead=clientChn.read(readBuff);
if(bytesRead==-1){
throw new IOException("远程已关闭");
}
while(bytesRead>0){
readBuff.flip();
out.write(readBuff.array(), 0, readBuff.limit());
readBuff.clear();
bytesRead=clientChn.read(readBuff);
}
}
key.interestOps(SelectionKey.OP_READ);
notifyRead(out.toByteArray());
}finally{
out.close();
}
}
@Override
public void handleWrite(SelectionKey key) throws IOException {
SocketChannel clientChn=(SocketChannel)key.channel();
ByteBuffer message=null;
while(!messageQueue.isEmpty()){
synchronized(lockObje){
if(!messageQueue.isEmpty()){
message=this.messageQueue.peek();
clientChn.write(message);
if(message.hasRemaining()){
break;
}
messageQueue.poll();
}
}
}
key.interestOps(SelectionKey.OP_READ);
}
/**
*
* 发送消息(此时不是真正发送,而是放在一个待发送的队列中)
*
* @param message 信息
* @since Ver 1.0
*/
@Override
public void sendMessage(byte[] message) {
messageQueue.add(ByteBuffer.wrap(message));
this.notifySend(message);
}
@Override
public void addReadListener(ReadListener rl) {
readList.add(rl);
}
@Override
public void addSendListener(SendListener sl) {
sendList.add(sl);
}
@Override
public void removeReadListener(ReadListener rl) {
readList.remove(rl);
}
@Override
public void removeSendListener(SendListener rl) {
sendList.remove(rl);
}
private void notifyRead(byte[] message) {
for(ReadListener rl : readList){
rl.handleEvent(message,this);
}
}
private void notifySend(byte[] message) {
for(SendListener rl : sendList){
rl.handleEvent(message);
}
}
}
socket的NIO操作的更多相关文章
- Java NIO 操作总结
问题: 1.Java NIO 出现大量CLOSE_WAIT或TIME_WAIT的端口无法释放 CLOSE_WAIT: 参考:http://my.oschina.net/geecoodeer/blog/ ...
- Java Socket(3): NIO
NIO采取通道(Channel)和缓冲区(Buffer)来传输和保存数据,它是非阻塞式的I/O,即在等待连接.读写数据(这些都是在一线程以客户端的程序中会阻塞线程的操作)的时候,程序也可以做其他事情, ...
- day 4 Socket 和 NIO Netty
Scoket通信--------这是一个例子,可以在这个例子的基础上进行相应的拓展,核心也是在多线程任务上进行修改 package cn.itcast.bigdata.socket; import j ...
- [ PyQt入门教程 ] PyQt+socket实现远程操作服务器
来需求了..干活啦.. 需求内容 部分时候由于缓存刷新.验证码显示不出来或者浏览器打不开或者打开速度很慢等原因,导致部分测试同事不想使用浏览器登录服务器执行命令.期望有小工具可以替代登录浏览器的操作, ...
- Socket网络通信——IO、NIO、AIO介绍以及区别
一 基本概念 Socket又称"套接字",应用程序通常通过"套接字"向网路发出请求或者应答网络请求. Socket和ServerSocket类位于java.ne ...
- NIO【同步非阻塞io模型】关于 NIO socket 的详细总结【Java客户端+Java服务端 + 业务层】【可以客户端间发消息】
1.前言 以前使用 websocket来实现双向通信,如今深入了解了 NIO 同步非阻塞io模型 , 优势是 处理效率很高,吞吐量巨大,能很快处理大文件,不仅可以 做 文件io操作, 还可以做sock ...
- 用socket操作redis
代码: $cmd = "*3\r\n$3\r\nSET\r\n$3\r\nfoo\r\n$3\r\nbar\r\n"; // set foo bar $socket = socke ...
- Java I/O之NIO Socket
PS:本文简单介绍下旧I/O和NIO下的Socket通讯,仅以UDP来示例. TCP/IP协议 首先简单回顾下TCP/IP协议 Application:应用程序:Socket:套接字:Host:主机: ...
- NIO vs. BIO
性能测试 BIO -- Blocking IO 即阻塞式IO NIO -- Non-Blocking IO, 即非阻塞式IO或异步IO 性能 -- 所谓的性能是指服务器响应客户端的能力,对于服务器我们 ...
随机推荐
- 了解SQL Server锁争用:NOLOCK 和 ROWLOCK 的秘密
关系型数据库,如SQL Server,使用锁来避免多用户修改数据时的并发冲突.当一组数据被某个用户锁定时,除非第一个用户结束修改并释放锁,否则其他用户就无法修改该组数据. 有些数据库,包括SQL Se ...
- HDU4725 The Shortest Path in Nya Graph dij
分析:对于每一层,原来n个点,然后扩展为原来的三倍,每一层扩展一个入点,一个出点,然后跑最短路 注:tmd我把一个n写成m了,然后wa了7次,我都要怀疑人生了 #include<cstdio&g ...
- FFT矩阵
举个例子: \[{F_4}=\left[{\begin{array}{*{20}{c}}1&1&1&1\\1&i&{-1}&{-i}\\1&{- ...
- Spark Pipe使用方法(外部程序调用方法)
写在前面: 1.我们使用的是Hadoop2.2.0,Spark 1.0. 2.这里使用的样例是经典的求pai程序来演示这个开发过程. 3.我们暂时使用java程序来开发,按照需要后面改用scala来开 ...
- 应用引擎BAE3.0介绍及百度BAE3.0支持并发多少
百度云BAE3.0的特点:1.支持本地程序迁移百度云应用引擎BAE3.0做了很多的改进,其实就是一句话,百度云开发在不断的进步.为了节省开发者的学习成本,百度云BAE3.0增加了轻量级虚拟机,使开发环 ...
- leptonica使用问题
想要使用leptonica编写一个图像处理程序,或者调用leptonica/prog下例子程序,出现类似错误: leptTest ./test Error in pixReadStreamJpeg: ...
- .Net设计模式_开篇
前言 其实以前看过两次设计模式,现在想来,几乎已经对设计模式没有任何印象,说明根本没有理解.或者说几乎不用,所以我除了单列.工厂外的设计模式几乎全部忘记了.最近需要写一个引擎,想用UML设计整体的架构 ...
- new Date() iso不支持兼容性问题
在IOS5以上版本(不包含IOS5)中的Safari浏览器能正确解释出Javascript中的 new Date('2013-10-21') 的日期对象. 但是在IOS5版本里面的Safari解释ne ...
- 虚机分配静态IP地址
在为虚机分配静态IP地址池的IP时,如果报错“没有从与 VMSubnet 相关的 IP 池中为虚拟网络适配器 9i [MAC: 001DD8B71C17] 分配 CA (客户地址)” 使用“动态获取” ...
- 用WebCollector爬取站点的图片
用WebCollector爬取整站图片,仅仅须要遍历整站页面.然后将URL为.jpg.gif的页面(文件)保存到本地就可以. 比如我们爬取一个美食站点,获取里面全部的图片: import cn.edu ...