NIO框架之MINA源码解析(五):NIO超级陷阱和使用同步IO与MINA通信
1、NIO超级陷阱
之所以说NIO超级陷阱,就是因为我在本系列开头的那句话,因为使用缺陷导致客户业务系统瘫痪。当然,我对这个问题进行了很深的追踪,包括对MINA源码的深入了解,但其实之所以会出现这个问题,它的根不是MINA的原因,而是JDK底层的问题。
JDK底层在实现nio时,为了能够唤醒等待在io上的线程,在windows平台使用了两个端口建立连接发消息实现。看如下代码:
- public class NIOTest {
- @Test
- public void test1(){
- final int MAXSIZE=1000;
- Selector [] sels = new Selector[ MAXSIZE];
- try{
- for( int i = 0 ;i< MAXSIZE ;++i ) {
- sels[i] = Selector.open();
- Thread.sleep(1000*10);
- }
- }catch( Exception ex ){
- ex.printStackTrace();
- }
- }
- }
也就是说每调用一次Selector.open(),就会占用两个随机可用端口,相互通信--这就是问题的根源。
当然对于我们的项目来说,这个问题的解决需要分两步,首先限制端口数到用户可接受范围内,这个比较容易,我们可用只调用一次Selector.open()方法即可,使用单利模式;第二步,因为Selector.open()方法每次都是使用的系统当前可用的随机端口,所以就有可能导致占用客户业务端口的情况,因此我们必须把Selector.open()所开的端口限制在一定范围内,最好可以通过代码指定使用哪几个端口,但是直到现在我们都没有找到,如果亲们有方法的话,很感谢能够告诉我....,。
Selector总结如下:
Windows下,Selector.open()会自己和自己建立两条TCP链接。不但消耗了两个TCP连接和端口,同时也消耗了文件描述符。
Linux下,Selector.open()会自己和自己建两条管道。同样消耗了两个系统的文件描述符。
来源于:http://blog.csdn.net/haoel/article/details/2224055
所以,现在我们现在就干脆用传统IO与MINA server通信吧。
2、使用同步IO与MINA通信
其实非常简单,唯一的难点就在于使用同步IO与MINA通信时,怎么编码和解码,还是看下面代码吧。
server 端(用的MINA)
- /**
- * 初始化设置,启动服务
- */
- public void startServer() {
- logger.debug("mina server start");
- int port = SysEvnVar.managerPort;
- logger.debug("manager端口号:" + port);
- IoAcceptor acceptor = new NioSocketAcceptor();
- /** 日志设置 */
- acceptor.getFilterChain().addLast("logger", new LoggingFilter());
- //编码与解码工厂,使用的技术就是JDK提供的ObjectOutStream
- ObjectSerializationCodecFactory objsCodec=new ObjectSerializationCodecFactory();
- objsCodec.setDecoderMaxObjectSize(DEFAULTDECODER);
- objsCodec.setEncoderMaxObjectSize(DEFAULTDECODER);
- /** 数据转换,编码设置 */
- acceptor.getFilterChain().addLast(
- "codec",
- new ProtocolCodecFilter(objsCodec));
- /** 设置业务处理类 */
- acceptor.setHandler(serverHandler);
- /** 设置buffer容量 */
- acceptor.getSessionConfig().setReadBufferSize(DEFAULTBUFFERSIZE);
- /** 设置空闲时间 */
- acceptor.getSessionConfig().setIdleTime(IdleStatus.BOTH_IDLE, DEFAULTIDLETIME);
- try {
- /** 绑定端口 */
- acceptor.bind(new InetSocketAddress(port));
- logger.info("mina server bind port ("+port+") sucess");
- } catch (IOException e) {
- logger.error("mina server bind port ("+port+") fail:" + e.getMessage(),e);
- }
- }
client端(使用同步IO)
- public void sendMessageToManager(HyRequest request) {
- Socket socket = null;
- BufferedInputStream inBuff = null;
- OutputStream outBuff = null;
- try {
- Integer port = Integer.parseInt(hyGlobalConfigureCacheImpl.get(
- "managerPort").toString());
- String host = hyGlobalConfigureCacheImpl.get("managerIp")
- .toString();
- socket = new Socket(host, port);
- inBuff = new BufferedInputStream(new DataInputStream(
- socket.getInputStream()));
- outBuff = socket.getOutputStream();
- String json = HyJsonUtil.reqeustToJsonStr(request);
- //编码开始,使MINA能够解析
- IoBuffer buf = IoBuffer.allocate(64);
- buf.setAutoExpand(true);
- buf.putObject(json);
- buf.flip();
- byte[] bytes =new byte[buf.limit()];
- buf.get(bytes);
- //编码结束,直接输出到客户端
- outBuff.write(bytes);
- outBuff.flush();
- if(request.getOperation() != null && !"quit".equals(request.getOperation())){
- String allreadstr = new String();
- byte[] b = new byte[512 * 1024];
- int len;
- byte[] tmp = new byte[0];
- if ((len = inBuff.read(b)) != -1) {
- tmp = new byte[len];
- System.arraycopy(b, 0, tmp, 0, len);
- }
- loger.debug("len:"+len);
- //解码开始,byte[]为MINA传过来的数据
- IoBuffer in = IoBuffer.wrap(tmp);
- Object obj = null;
- try {
- loger.debug("in:"+in);
- obj = in.getObject();
- //解码结束
- loger.debug("parse success");
- } catch (Exception e) {
- // TODO Auto-generated catch block
- e.printStackTrace();
- loger.debug("go to exception");
- loger.warn("nio parse exception",e);
- obj = "exception";
- }
- allreadstr = (String) obj;
- loger.info("receive message from "
- + socket.getRemoteSocketAddress().toString() + ",message:"
- + allreadstr);
- // 请求消息转换为HyResponse对象
- HyResponse response = HyJsonUtil.getHyResponse(allreadstr);
- HyRequest hyrequest = new HyRequest();
- hyResponseDispatcher = hyClientHandler.getHyResponseDispatcher();
- hyResponseDispatcher.responseProcess(hyrequest, response);
- if (hyrequest.getOperation() != null) {
- sendMessageToManager2(hyrequest);
- }
- }
- } catch (Exception e) {
- e.printStackTrace();
- } finally {
- try {
- if(null != outBuff)
- outBuff.close();
- } catch (IOException e1) {
- // TODO Auto-generated catch block
- e1.printStackTrace();
- }
- try {
- if(null != inBuff)
- inBuff.close();
- } catch (IOException e1) {
- // TODO Auto-generated catch block
- e1.printStackTrace();
- }
- if (socket != null) {
- try {
- socket.close();
- } catch (IOException e) {
- }
- }
- }
- }
看代码里面的注释,怎么编码和解码,其实我是直接看MINA源码,把MINA的编码和解码方法工具类直接拷到了我们的client端,这样就轻松的实现使用同步IO和MINA进行通信。
NIO框架之MINA源码解析(五):NIO超级陷阱和使用同步IO与MINA通信的更多相关文章
- Celery 源码解析五: 远程控制管理
今天要聊的话题可能被大家关注得不过,但是对于 Celery 来说确实很有用的功能,曾经我在工作中遇到这类情况,就是我们将所有的任务都放在同一个队列里面,然后有一天突然某个同学的代码写得不对,导致大量的 ...
- dubbo源码解析五 --- 集群容错架构设计与原理分析
欢迎来我的 Star Followers 后期后继续更新Dubbo别的文章 Dubbo 源码分析系列之一环境搭建 博客园 Dubbo 入门之二 --- 项目结构解析 博客园 Dubbo 源码分析系列之 ...
- ReactiveCocoa源码解析(五) SignalProtocol的observe()、Map、Filter延展实现
上篇博客我们对Signal的基本实现以及Signal的面向协议扩展进行了介绍, 详细内容请移步于<Signal中的静态属性静态方法以及面向协议扩展>.并且聊了Signal的所有的g功能扩展 ...
- ReactiveSwift源码解析(五) SignalProtocol的observe()、Map、Filter延展实现
上篇博客我们对Signal的基本实现以及Signal的面向协议扩展进行了介绍, 详细内容请移步于<Signal中的静态属性静态方法以及面向协议扩展>.并且聊了Signal的所有的g功能扩展 ...
- iOS即时通讯之CocoaAsyncSocket源码解析五
接上篇:iOS即时通讯之CocoaAsyncSocket源码解析四 原文 前言: 本文为CocoaAsyncSocket Read篇终,将重点涉及该框架是如何利用缓冲区对数据进行读取. ...
- Spring 源码解析之DispatcherServlet源码解析(五)
spring的整个请求流程都是围绕着DispatcherServlet进行的 类结构图 根据类的结构来说DispatcherServlet本身也是继承了HttpServlet的,所有的请求都是根据这一 ...
- Mybaits 源码解析 (十)----- 全网最详细,没有之一:Spring-Mybatis框架使用与源码解析
在前面几篇文章中我们主要分析了Mybatis的单独使用,在实际在常规项目开发中,大部分都会使用mybatis与Spring结合起来使用,毕竟现在不用Spring开发的项目实在太少了.本篇文章便来介绍下 ...
- NIO框架之MINA源码解析(四):粘包与断包处理及编码与解码
1.粘包与段包 粘包:指TCP协议中,发送方发送的若干包数据到接收方接收时粘成一包,从接收缓冲区看,后一包数据的头紧接着前一包数据的尾.造成的可能原因: 发送端需要等缓冲区满才发送出去,造成粘包 接收 ...
- NIO框架之MINA源码解析(转)
http://blog.csdn.net/column/details/nio-mina-source.html http://blog.csdn.net/chaofanwei/article/det ...
随机推荐
- 给Linux服务器设置共享文件目录
通过samba软件 :yum install samba 修改配置文件:vi /etc/samba/smb.conf 添加 [WORKSPACE] comment = workspace ...
- django实现api跨域请求访问
第一步:安装 django-cors-headers pip install django-cors-headers 第二步:配置settings.py文件 --------------------- ...
- web项目【3】
前端代码: <%@ Page Language="C#" AutoEventWireup="true" CodeFile="Default2.a ...
- apache kafka系列之在zookeeper中存储结构
1.topic注册信息 /brokers/topics/[topic] : 存储某个topic的partitions所有分配信息 Schema: { "version": ...
- ESB雏形 -- 项目企业服务总线初始
今天要厚着脸皮给大家推荐一个自己做的通信中间件——ServiceAnt,目前已经在我们团队的两个产品线上投入了使用. ServiceAnt是什么 它最初的定位是ESB(企业服务总线),但目前还没有达到 ...
- BTrace学习总结
一.简介: 在生产环境中经常遇到格式各样的问题,如OOM或者莫名其妙的进程死掉.一般情况下是通过修改程序,添加打印日志:然后重新发布程序来完成.然而,这不仅麻烦,而且带来很多不可控的因素.有没有一种方 ...
- shiro学习笔记
一.概念: shiro是apache旗下一个开源框架,它将软件系统的安全认证相关的功能抽取出来,实现用户身份认证.权限授权.加密.会话管理等功能,组成了一个通用的安全认证框架. (一)shiro的功能 ...
- Docker和Rancher
Docker打包流程: Dockerfile文件和要打包docker的文件放在同级目录下: 1. docker build -t proj:proj-app:0.0.1 返回tagXXX 2. doc ...
- [shell]find命令
find ./ -mtime 18 -exec ls -l {} \; 该范例中特殊的地方有 {} 以及 \; 还有 -exec 这个关键字,这些东西的意义为: {} 代表的是『由 find 找到的内 ...
- dojo DataGrid实现表格数据编辑的解决方案
在官网上看见的DataGrid编辑非常简单,但我实现的时候总是出现问题.经过N久的摸索,终于搞定了,期间出现了多处困难,下面说些解决办法的流程. 我实现的表格在页面加载时是不显示数据,只有通过表单像服 ...