MINA开发-1-服务器端开发

MINA服务端的应用应用体系结构

基本上,服务器监听端口传入请求,处理它们并发送回复。它还为每个客户端创建和处理会话(无论我们是基于TCP或者UDP的时候)。


Server_arch.png
  • IOAcceptor在网络上监听传入的连接/数据包
  • 对于新链接,将会创建一个新的会话,并在该会话中处理来自IP地址/端口组合的所有后续请求
  • 为会话接收的所有数据包将按照图中所示的方式遍历过滤器。过滤器可用于修改数据包的内容(如转换为对象,添加/删除信息等)。PacketEncoder/Decoder对于将原始字节转换为高级对象,非常有用。
  • 最后数据包或者转换成对象的数据将加载到IOHandler中,IOHandler可以非常完美的处理业务需求。

示例TCP服务器

接下来将引导您完成构建基于MINA的程序的过程,逐步构建一个时间服务器。开发服务器端需要以下几个先决条件:
 1.MINA 2.x Core
 2.JDK1.5或者更高
 3.SLF4J1.3.0或者更高(注1中的下载文件中已经包含了对应的包)
    - Log4J 1.2 的用户需要的包:slf4j-api.jar,slf4j-log4j12.jar,和Log4J 1.2.x
    - Log4J 1.3的用户需要的包:slf4j-api.jar,slf4j-log4j13.jar,和Log4J 1.3.x
    - java.util.logging的用户需要的包:slf4j-api.jar和slf4j-jdk14.jar,
    - 注:请确保你使用了正确的slf4j-*.jar匹配你的日志框架。
实际上, slf4j-log4j12.jar 和log4j-1.3.x.jar 不能一起使用,并且会发生故障。

开始服务器端代码

首先创建java project项目,和普通的创建方式一致,创建libs文件夹将需要的包复制进去:


项目结构.png

接下来创建对应的服务类:

package com.fmblzf.minaservice;
/**
 * 
 * @Copy:2017-fmblzf 
 * @ProjectName:MINA_SERVER 
 * @ClassDecription:     
 * @ClassName:com.fmblzf.minaservice.MinaServerMain      
 * @Creator:zhaofeng     
 * @CreatTime:2017年5月27日 上午11:50:55  
 * @FixPerson:fmblzf
 * @FixTime:2017年5月27日 上午11:50:55    
 * @Tag: 
 * @version V1.0 
 *
 */
public class MinaServerMain {
        //设置端口号
    private static final int PORT = 9321;
    public static void main(String[] args) {

    }
}

首先,我们需要一个将用于侦听传入连接的对象。由于该程序将基于TCP/IP,我们将在我们的程序中添加一个SocketAcceptor,我们创建对应的NioSocketAcceptor对象,并且将其绑定到PORT端口上。

    IoAcceptor ioAcceptor = new NioSocketAcceptor();
    ioAcceptor.bind(new InetSocketAddress(PORT));

接下来设置过滤器,配置Session,以及设置处理对象

public static void main(String[] args) throws IOException {
        //首先,添加对应的网络监听对象,等待处理连接和回复消息
        IoAcceptor ioAcceptor = new NioSocketAcceptor();
        //添加过滤器
        ioAcceptor.getFilterChain().addLast("logger",new LoggingFilter());//设置日志管理过滤器
        ioAcceptor.getFilterChain().addLast("codec", new ProtocolCodecFilter(new ObjectSerializationCodecFactory()));//设置字节处理过滤器
        //添加IOHandler
        ioAcceptor.setHandler(new MainServerHandler());
        //添加Session配置       
        ioAcceptor.getSessionConfig().setReadBufferSize(2048);//设置读缓存区的大小       
        ioAcceptor.getSessionConfig().setIdleTime(IdleStatus.BOTH_IDLE, 10);//设置读写的空闲时间都是10秒
        ioAcceptor.bind(new InetSocketAddress(PORT));

    }

创建自定义的IoHandler对象

/**
     * 逻辑处理类
     * @Copy:2017-fmblzf 
     * @ProjectName:MINA_SERVER 
     *   
     * @ClassDecription:     
     * @ClassName:com.fmblzf.minaservice.MainServerHandler   
     * @Creator:fmblzf   
     * @CreatTime:2017年5月27日 下午12:41:08  
     * @FixPerson:fmblzf  
     * @FixTime:2017年5月27日 下午12:41:08    
     * @Tag: 
     * @version V1.0 
     *
     */
    private static class MainServerHandler extends IoHandlerAdapter{
        /**
         * 接收到消息
         * @see org.apache.mina.core.service.IoHandlerAdapter#messageReceived(org.apache.mina.core.session.IoSession, java.lang.Object) 
         *
         */
        @Override
        public void messageReceived(IoSession session, Object message)
                throws Exception {
            String str = message.toString();
            if( str.trim().equalsIgnoreCase("quit") ) {
                session.closeNow();
                return;
            }
            Date date = new Date();
            session.write( date.toString() );
            System.out.println("Message written...");
        }
        
        @Override
        public void sessionIdle(IoSession session, IdleStatus status)
                throws Exception {
            System.out.println( "IDLE " + session.getIdleCount( status ));
        }
        /**
         * 发送消息给客户端
         * @see org.apache.mina.core.service.IoHandlerAdapter#messageSent(org.apache.mina.core.session.IoSession, java.lang.Object) 
         *
         */
        @Override
        public void messageSent(IoSession session, Object message)
                throws Exception {
        }
        @Override
        public void sessionCreated(IoSession session) throws Exception {
        }
        @Override
        public void sessionOpened(IoSession session) throws Exception {
        }
        @Override
        public void sessionClosed(IoSession session) throws Exception {
        }
    }

示例UDP服务器

因为MINA的统一API接口,所以UDP和TCP在结构上是一致的,只有IoAcceptor对应的实现类是不一致,所以接下来我们看一看UDP创建的核心代码:


    /**
     * 创建UDP连接
     * @Title: createUdp 
     * @Description: TODO 
     * @throws IOException 
     *
     */
    private static void createUdp() throws IOException {
        IoAcceptor ioAcceptor = new NioDatagramAcceptor();
        // 添加过滤器
        ioAcceptor.getFilterChain().addLast("logger", new LoggingFilter());// 设置日志管理过滤器
        ioAcceptor.getFilterChain().addLast("codec",
                new ProtocolCodecFilter(new ObjectSerializationCodecFactory()));// 设置字节处理过滤器
        // 添加IOHandler
        ioAcceptor.setHandler(new MainServerHandler());
        // 添加Session配置
        ioAcceptor.getSessionConfig().setReadBufferSize(2048);// 设置读缓存区的大小
        ioAcceptor.getSessionConfig().setIdleTime(IdleStatus.BOTH_IDLE, 10);// 设置读写的空闲时间都是10秒
        ioAcceptor.bind(new InetSocketAddress(PORT+1));
    }

从代码我们可以看见,UDP的IoAcceptor的实现类是NioDatagramAcceptor,其他的结构编写都是一致。

至此我们的TCP/UDP的核心代码已经完成,至于其他的衍生部分,感兴趣的朋友可以参考官网来接着学习。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,029评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,395评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 157,570评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,535评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,650评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,850评论 1 290
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,006评论 3 408
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,747评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,207评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,536评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,683评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,342评论 4 330
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,964评论 3 315
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,772评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,004评论 1 266
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,401评论 2 360
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,566评论 2 349

推荐阅读更多精彩内容

  • spring官方文档:http://docs.spring.io/spring/docs/current/spri...
    牛马风情阅读 1,650评论 0 3
  • Spring Boot 参考指南 介绍 转载自:https://www.gitbook.com/book/qbgb...
    毛宇鹏阅读 46,773评论 6 342
  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,633评论 18 139
  • 从三月份找实习到现在,面了一些公司,挂了不少,但最终还是拿到小米、百度、阿里、京东、新浪、CVTE、乐视家的研发岗...
    时芥蓝阅读 42,213评论 11 349
  • 简介 用简单的话来定义tcpdump,就是:dump the traffic on a network,根据使用者...
    保川阅读 5,946评论 1 13