/**
* 非阻塞IO多线线程服务端
* 当一个任务进入多线程,这个任务线程需要处理接收信息、发送信息、因而发生I/O阻塞问题
* 利用selector可以实现异步
*
*/
public class EchoServer02 {
//轮询器,处理IO阻塞问题
private Selector selector = null;
private ServerSocketChannel serverSocketChannel = null;
private int port = ;
private Charset charset = Charset.forName("GBK");//编码方式 public EchoServer02() throws IOException{
//创建一个Selector对象
selector = Selector.open();
//这个方法没有与任何本地端口绑定,并且处于阻塞模式;
serverSocketChannel = ServerSocketChannel.open();
serverSocketChannel.socket().setReuseAddress(true);
//使ServerSocketChannel工作于非阻塞模式
serverSocketChannel.configureBlocking(false);// IO异步处理
//把服务器与本地端口绑定
serverSocketChannel.socket().bind(new InetSocketAddress(port));//绑定服务器端口
System.out.println("服务器已启动");
} public void service() throws Exception{
/*SeverSocketChannel或Socket类通过register()方法向Selector注册事件时,
register()方法会创建一个SelectionKey对象,
这个SelectionKey对象是用来跟踪注册事件的句柄。
在SelectionKey对象的有效时间,Selector会一直监控与SelectionKey对象相关的事件,
如果事件发生,就会把SelectionKey对象加入seleected-keys集合中。*/
//将ServerSocketChannel注册到Selector上
//只要ServerSocketChannel及SocketChannel向Selector注册了特定的事件,Selector就会监控这些事件是否发送
//SelecitonKey.OP_ACCEPT:接收连接就绪事件,表示服务器监听到了客户连接,服务器可以接收这个链接了。常量值为16
//这个客户SocketChannel会被Selector监控到
serverSocketChannel.register(selector, SelectionKey.OP_ACCEPT);
//获取Selector中的SelectionKey数量
while(selector.select() > ){//第一层循环
//相关事件已经被Selector捕获的SelectionKey的集合。
Set readKeys = selector.selectedKeys();
Iterator iterator = readKeys.iterator();
while(iterator.hasNext()){//第二层循环
SelectionKey key = null;
try{//处理SelectionKey
key = (SelectionKey)iterator.next();//取出一个SelectionKey
//把SelectionKey从Selector的selected-集合中删除
iterator.remove();
if(key.isAcceptable()){//处理连接就绪事件
//获得与SelectionKey关联的ServerSocketChannel
ServerSocketChannel serverSocketChannel = (ServerSocketChannel) key.channel();
//获得与客户连接的SocketChannel,这个SocketChannel默认情况是阻塞的
SocketChannel socketChannel = serverSocketChannel.accept();
System.out.println("接收到客户的连接,来自:"+socketChannel.socket().getInetAddress()
+":"+socketChannel.socket().getPort());
//把SocketChannel设置为非阻塞模式,
socketChannel.configureBlocking(false);
//创建一个用于存放用户发送来的数据的缓冲区
ByteBuffer buffer = ByteBuffer.allocate();
//把SocketChannel向Selector注册就读事件和就绪事件,且关联了一个buffer附件
socketChannel.register(selector, SelectionKey.OP_READ|SelectionKey.OP_WRITE,buffer);
}
if(key.isReadable()){//处理读就绪事件
receive(key);
}
if(key.isWritable()){//处理写就绪事件
send(key);
}
}catch (Exception e) {
e.printStackTrace();
try {
if(key != null){
//使这个Seleciton失效
//使得Selector不再监控这个SelectionKey感兴趣的事件
key.cancel();
key.channel().close();//关闭这个SelectionKey关联的SocketChannel
}
} catch (Exception e2) {
e2.printStackTrace();
}
}
}
}
} public void send(SelectionKey key) throws IOException{
//获得与SelectionKey关联的ByteBuffer
ByteBuffer buffer = (ByteBuffer) key.attachment();
//获得与SelectionKey关联的SocketChannel
SocketChannel socketChannel = (SocketChannel) key.channel();
//把极限设为位置,把位置设为0
buffer.flip();
//按照GBK编码,把buffer中的字节转换为字符串
String data = decode(buffer);
//如果还没有读到一行数据就返回
if(data.indexOf("\r\n") == -){
return ;
}
//截取一行数据
String outputData = data.substring(, data.indexOf("\n")+);
System.out.println(outputData);
//把输出的字符串安装GBK编码,转换为字节,把它放入outputBuffer
ByteBuffer outputBuffer = encode("ehco:" + outputData);
//输出outputBuffer中所有的字节
while(outputBuffer.hasRemaining()){
socketChannel.write(outputBuffer);
//把outputData字符串按照GBK编码,转换为字节 ,把它放入ByteBuffer中
ByteBuffer temp = encode(outputData);
//把buffer的位置设为temp的极限
buffer.position(temp.limit());
//删除buffer中已经处理的数据
buffer.compact();
//如果已经输出了字符串"bye\r\n",就使SelectionKey失效,并关闭SocketChannel
if(outputData.equals("bye\r\n")){
key.cancel();
socketChannel.close();
System.out.println("关闭与客户的连接");
}
}
}
/**
* receive()方法把读入的数据都放在一个ByteBuffer中,
* send()方法就从这个ByteBuffer中取出数据
* 如果ByteBuffer中还没有一行字符串,就什么不做,直接退出send()方法
* @param key
* @throws IOException
*/
public void receive(SelectionKey key) throws IOException{
//获得与SelectionKey关联的附件
ByteBuffer buffer = (ByteBuffer) key.attachment();
//获得与SelectionKey关联的SocketChannel
SocketChannel socketChannel = (SocketChannel) key.channel();
//创建一个ByteBuffer,用于存放读到的数据
ByteBuffer readBuff = ByteBuffer.allocate();
socketChannel.read(readBuff);
readBuff.flip();
//把buffer的极限设为容量
buffer.limit(buffer.capacity());
//把readBuff中内容拷贝到buffer中,
//假设buffer容量足够大,不会出现缓冲区溢出异常
buffer.put(readBuff);
} public String decode(ByteBuffer buffer){//解码,将字节转换为字符串的过程
CharBuffer charBuffer = charset.decode(buffer);
return charBuffer.toString();
} public ByteBuffer encode(String str){//编码,将字符串转换为字节
return charset.encode(str);
} public static void main(String[] args) throws Exception{
EchoServer02 server02 = new EchoServer02();
server02.service();
} }

java 服务端I/O非阻塞实现05的更多相关文章

  1. Java 服务端监控方案(四. Java 篇)

    http://jerrypeng.me/2014/08/08/server-side-java-monitoring-java/ 这个漫长的系列文章今天要迎来最后一篇了,也是真正与 Java 有关的部 ...

  2. 那些年,我们见过的 Java 服务端“问题”

    导读 明代著名的心学集大成者王阳明先生在<传习录>中有云: 道无精粗,人之所见有精粗.如这一间房,人初进来,只见一个大规模如此.处久,便柱壁之类,一一看得明白.再久,如柱上有些文藻,细细都 ...

  3. 俯瞰 Java 服务端开发

    原文首发于 github ,欢迎 star . Java 服务端开发是一个非常宽广的领域,要概括其全貌,即使是几本书也讲不完,该文将会提到许多的技术及工具,但不会深入去讲解,旨在以一个俯瞰的视角去探寻 ...

  4. Java服务端性能优化

    <Java程序性能优化>说性能优化包含五个层次:设计调优.代码调优.JVM调优.数据库调优.操作系统调优. 常用的几个代码优化方案: 使用单例 对于IO处理.数据库连接.配置文件解析加载等 ...

  5. java服务端集成极光消息推送--详细开发步骤

    1.极光推送账号准备 要使用极光消息推送必须先在官方网站上注册账号,并添加应用. 产品介绍:https://docs.jiguang.cn/jpush/guideline/intro/ 注册开发者账号 ...

  6. RPC学习--C#使用Thrift简介,C#客户端和Java服务端相互交互

    本文主要介绍两部分内容: C#中使用Thrift简介 用Java创建一个服务端,用C#创建一个客户端通过thrift与其交互. 用纯C#实现Client和Server C#服务端,Java客户端 其中 ...

  7. ajax跨域请求,页面和java服务端的写法

    方法一(jsonp): 页面ajax请求的写法: $.ajax({ type : "get", async : false, cache : false, url : " ...

  8. Flex通信-Java服务端通信实例

    转自:http://blessht.iteye.com/blog/1132934Flex与Java通信的方式有很多种,比较常用的有以下方式: WebService:一种跨语言的在线服务,只要用特定语言 ...

  9. “快的打车”创始人陈伟星的新项目招人啦,高薪急招Java服务端/Android/Ios 客户端研发工程师/ mysql DBA/ app市场推广专家,欢迎大家加入我们的团队! - V2EX

    "快的打车"创始人陈伟星的新项目招人啦,高薪急招Java服务端/Android/Ios 客户端研发工程师/ mysql DBA/ app市场推广专家,欢迎大家加入我们的团队! - ...

随机推荐

  1. DocX开源WORD操作组件的学习系列三

    DocX学习系列 DocX开源WORD操作组件的学习系列一 : http://www.cnblogs.com/zhaojiedi1992/p/zhaojiedi_sharp_001_docx1.htm ...

  2. RDIFramework.NET ━ .NET快速信息化系统开发框架 V3.2-模块管理按子系统进行分类管理

    在RDIFramework.NET以往的框架中,模块管理界面展示了整个框架所管理的所有模块,如果系统过多,达几十个甚至上百个子系统时,管理起来就非常的麻烦,不光加载效率会很低,页面展示也会很不友好.框 ...

  3. 痞子衡嵌入式:ARM Cortex-M内核那些事(2)- 第一款微控制器

    大家好,我是痞子衡,是正经搞技术的痞子.今天痞子衡给大家介绍的是第一款Cortex-M微控制器. 1.天生荣耀:ARM Cortex-M处理器由来 ARM公司自2004年推出ARMv7内核架构时,摒弃 ...

  4. 前端异步技术之Promise

    前言 从事前端的朋友或多或少的接触过Promise,当代码中回调函数层级过多你就会发现Promise异步编程的魅力,相信此文一定能帮你排忧解惑! Promise概念 Promise是JS异步编程中的重 ...

  5. 基于Dockerfile镜像制作的基本操作

    一.使用Dockerfile制作镜像 前面的博客中已经介绍了如何基于容器制作镜像,此方法的原理是使用一个正在运行的容器,根据生产所需进行配置更改等操作后,使其满足生产环境,再将这个容器打包制作为镜像, ...

  6. 解读经典《C#高级编程》泛型 页114-122.章4

    前言 本章节开始讲解泛型..Net从2.0开始支持泛型,泛型不仅是C#的一部分,也与IL代码紧密集成.所以C#中泛型的实现非常优雅.相对于C#,Java是后期引入的泛型,受限于最初的设计架构,就实现的 ...

  7. (摘)老司机也必须掌握的MySQL优化指南

    当 MySQL 单表记录数过大时,增删改查性能都会急剧下降,本文会提供一些优化参考,大家可以参考以下步骤来优化. 单表优化 除非单表数据未来会一直不断上涨,否则不要一开始就考虑拆分,拆分会带来逻辑.部 ...

  8. [Go] golang连接redis测试

    go-redis的使用1.下载代码到GOPATH环境变量指定的目录比如我的是进入目录D:\golang\code\src\github.com\go-redis , 执行git clone https ...

  9. 20190322-a标签、img标签、三列表、特殊字符实体、表格

    目录 1.a标签 a标签的属性 锚点 2.img标签 img标签的属性 图像热区 3.三列表 有序列表(Ordered List)     ol>li 无序列表(Unordered List)  ...

  10. vuex2中使用mapGetters/mapActions报错解决方法

    解决方案 可以安装整个stage2的预置器或者安装 Object Rest Operator 的babel插件 babel-plugin-transform-object-rest-spread . ...