前提:
1、MINA 2.0.7 Core
2、JDK 1.5 或更高版本
3、[SLF4J|http://www.slf4j.org/] 1.3.0 orgreater
1)Log4J 1.2 users: slf4j-api.jar, slf4j-log4j12.jar, and Log4J 1.2.x
2)Log4J 1.3 users: slf4j-api.jar,slf4j-log4j13.jar, and Log4J 1.3.x
3)java.util.logging users: slf4j-api.jar and slf4j-jdk14.jar
注意:确保slf4j-*.jar和框架相配合,比如 slf4j-log4j12.jar and log4j-1.3.x.jar 不能同时使用,否则会失灵。
第一步:搭建项目
1、建立项目Java Project:MinaTimeServer
2、解压apache-mina-2.0.7-bin.zip,进入文件夹apache-mina-2.0.7,将dist目录下的mina-core-2.0.7.jar拷贝到项目中,java需要Add to build Path,JavaWeb直接添加到Web-Inf/lib目录下,会自动构建到路径。
3、从http://www.slf4j.org/download.html下载slf4j-1.7.5.zip,解压,将里面slf4j-log4j12-1.7.5.jar同样方式添加到项目中。
4、从http://www.apache.org/dyn/closer.cgi/logging/log4j/1.2.17/log4j-1.2.17.zip 下载log4j-1.2.17.zip,解压,将log4j-1.2.17.jar同样方式添加到项目中。
并在项目的src目录下,或者新建conf源文件夹目录,建立log4j.properties(必须叫此名):
log4j.rootLogger=DEBUG,console log4j.logger.sys=DEGUG,system log4j.logger.busi=INFO,business log4j.appender.system=org.apache.log4j.DailyRollingFileAppender log4j.appender.system.File=rfm-sys log4j.appender.system.DatePattern=yyyy-MM-dd'.log' log4j.appender.system.encoding=GBK log4j.appender.system.layout=org.apache.log4j.PatternLayout log4j.appender.system.layout.ConversionPattern=[%5p][%d{yyyy-MM-ddHH:mm:ss.SSS}]-%m%n log4j.appender.business=org.apache.log4j.DailyRollingFileAppender log4j.appender.business.File=rfm-busi log4j.appender.business.DatePattern=yyyy-MM-dd'.log' log4j.appender.business.encoding=GBK log4j.appender.business.layout=org.apache.log4j.PatternLayout log4j.appender.business.layout.ConversionPattern=[%5p][%d{yyyy-MM-ddHH:mm:ss.SSS}]-%m%n log4j.appender.console=org.apache.log4j.ConsoleAppender log4j.appender.console.encoding=GBK log4j.appender.console.layout=org.apache.log4j.PatternLayout log4j.appender.console.layout.ConversionPattern=[%5p][%d{yyyy-MM-ddHH\:mm\:ss.SSS}]-%m%n
第二步:建立时间服务器
1、 建立TimeServerHandler.java类:
package zxn.mina; importjava.text.SimpleDateFormat; import java.util.Date; importorg.apache.mina.core.service.IoHandlerAdapter; importorg.apache.mina.core.session.IdleStatus; importorg.apache.mina.core.session.IoSession; /** * 说明:定义为客户端的连接与请求服务的处理器,处理MINA构建的网络应用程序的主要业务逻辑(因为这里处理所有的客户端请求)。 * 功能:返回当前的准确时间 * @author Administrator * 注意:该处理器类必须实现IoHandler接口,在这我们实现IoHandlerAdapter,它采用适配器模式,简化了代码书写 * 使用:需要将其配置到NioSocketAcceptor才能发挥作用 */ publicclass TimeServerHandler extends IoHandlerAdapter{ // 处理器必须覆盖该方法,用于处理处理器处理远程连接时发生的异常;否则,异常不能被正确报出 @Override publicvoid exceptionCaught( IoSession session,Throwable cause ) throws Exception { // 该方法仅仅简单地打印出异常,然后关闭当前和客户端连接的会话,这是大多数程序的标准做法,除非处理器能从异常中恢复 cause.printStackTrace(); } // 接收客户端的数据,并做出自己想要实现的处理,当前是返回当前时间 // 第二个参数若在write()方法中没有指定协议编解码器,则默认是是IoBuffer类型,写出的数据类型也必须是IoBuffer类型 @Override publicvoid messageReceived( IoSession session, Objectmessage ) throws Exception { String str = message.toString(); if( str.trim().equalsIgnoreCase("bye") ) { session.write("bye!"); session.close(true); return; } SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss SSS"); session.write( "the current time is: "+sdf.format(new Date()) ); System.out.println(""); } // 一旦会话空闲时间超过了call acceptor.getSessionConfig().setIdleTime( IdleStatus.BOTH_IDLE, 10 )定义的时间则会被调用 @Override publicvoid sessionIdle(IoSession session, IdleStatus status ) throws Exception { System.out.println( "会话空闲段 " + session.getIdleCount( status )); } }
2、 建立MinaTimeServer.java类:
package zxn.mina; importjava.io.IOException; import java.net.InetSocketAddress; importjava.nio.charset.Charset; importorg.apache.mina.core.service.IoAcceptor; importorg.apache.mina.core.session.IdleStatus; importorg.apache.mina.filter.codec.ProtocolCodecFilter; importorg.apache.mina.filter.codec.textline.TextLineCodecFactory; importorg.apache.mina.filter.logging.LoggingFilter; importorg.apache.mina.transport.socket.nio.NioSocketAcceptor; /** * 基于TCP/IP,构建时间服务器 * @author Administrator * */ publicclass MinaTimeServer { // 定义监听的端口号 privatestaticfinalintPORT = 8088; publicstaticvoid main( String[]args ) throws IOException { // 定义异步Socket接收器 IoAcceptoracceptor = new NioSocketAcceptor(); // 在过滤连末尾添加日志过滤器,记录新创建的会话,发送接收的消息,以及会话的关闭 acceptor.getFilterChain().addLast( "logger", new LoggingFilter() ); // 在过滤连末尾添加编码解码过滤器,用于将二进制数据或者自定义结构的数据转化为消息对象(解码),反之为编码。 acceptor.getFilterChain().addLast( "codec", newProtocolCodecFilter( new TextLineCodecFactory( Charset.forName( "UTF-8" )))); // 设置处理连接的处理器 acceptor.setHandler( newTimeServerHandler() ); // 设置底层操作系统为每次传入的数据分配多大的空间 acceptor.getSessionConfig().setReadBufferSize( 2048 ); // 设置何时检查闲置会话,第一个参数表示决定会话为闲置的依据(服务端发送消息,客户端发送消息,两者都算),第二个参数表示在这个时间段内,前面依据之任一必须发生,否则被认为是空闲会话。 acceptor.getSessionConfig().setIdleTime( IdleStatus.BOTH_IDLE, 10 ); // 将接收器绑定到指定本地端口 acceptor.bind( new InetSocketAddress(PORT) ); } }
3、 项目结构如下:
第三步:测试
1、进入cmd控制行,输入telnet127.0.0.1 9123
2、回车
4、 再回车,得到服务器返回的时间服务:
5、 输入bye,则中断和服务器的连接: