下午那篇博客我們講到了Mina的客戶端的開發,如果還有沒看過的同學可以看一下,我是傳送門。現在,我們來學習一下,Mina的服務器的開發。
一、首先看一下,我的服務器的代碼圖片:
服務器代碼我是在MyEclipse下寫的。
二、服務器的整體思路:(同客戶端差不多)
-
-
- 首先,產生一個socket接收對象(SocketAcceptor),用於接收客戶端的連接請求;
- 然后,對這個接收器添加我們的I/O過濾器(SSL加密、日志過濾器、編碼過濾器等,這里注意,如果添加SSL過濾器,那么一定要第一個添加,否則無法對數據加密);
- 接下來,為連接設置I/O處理器,顧名思義就是處理接收到的消息(這里我們只能設置一個處理器,如果有設置多個,那么默認進入到最后一個I/O處理器中進行處理);
- 最后,將服務器綁定到某端口(如:3456,最好是1024以上,因為1024以下的端口系統占用)。
-
三、正式編碼
這里我同樣展示幾個比較重要的類來詳細說明一下:
-
-
- MinaServer.Java
1 package com.mina.example; 2 3 import java.io.IOException; 4 import java.net.InetSocketAddress; 5 6 import org.apache.mina.core.filterchain.DefaultIoFilterChainBuilder; 7 import org.apache.mina.core.session.IdleStatus; 8 import org.apache.mina.filter.codec.ProtocolCodecFilter; 9 import org.apache.mina.filter.logging.LoggingFilter; 10 import org.apache.mina.filter.ssl.SslFilter; 11 import org.apache.mina.transport.socket.SocketAcceptor; 12 import org.apache.mina.transport.socket.nio.NioSocketAcceptor; 13 14 import com.mina.charset.CharsetFactory; 15 import com.mina.hanlder.MsgHanler; 16 import com.mina.ssl.SSLContextGenerator; 17 18 /** 19 * <pre> 20 * Project Name:MinaServer 21 * Package:com.mina.example 22 * FileName:MinaServer.java 23 * Purpose:服務器 24 * Create Time: 2014-8-19 下午4:59:55 25 * Create Specification: 26 * Modified Time: 27 * Modified by: 28 * Modified Specification: 29 * Version: 1.0 30 * </pre> 31 * 32 * @author myp 33 */ 34 public class MinaServer { 35 36 private SocketAcceptor acceptor; 37 38 public MinaServer() { 39 /* 40 * 1.創建一個socket連接,連接到服務器 41 */ 42 acceptor = new NioSocketAcceptor(); 43 } 44 45 public boolean start() { 46 /* 47 * 獲取過濾器鏈,用於添加過濾器 48 */ 49 DefaultIoFilterChainBuilder filterChain = acceptor.getFilterChain(); 50 51 /* 52 * 2.為連接添加過濾器,SSL、日志、編碼過濾器 53 */ 54 // SSLContextGenerator是我們自己寫的一個SSL上下文產生器,稍后會講到 55 SslFilter sslFilter = new SslFilter( 56 new SSLContextGenerator().getSslContext()); 57 // a.ssl過濾器,這個一定要第一個添加,否則數據不會進行加密 58 filterChain.addLast("sslFilter", sslFilter); 59 System.out.println("SSL support is added.."); 60 // b.添加日志過濾器 61 filterChain.addLast("loger", new LoggingFilter()); 62 // c.添加字符的編碼過濾器 63 filterChain.addLast("codec", new ProtocolCodecFilter( 64 new CharsetFactory())); 65 66 /* 67 * 3.設置消息處理器,用於處理接收到的消息 68 */ 69 acceptor.setHandler(new MsgHanler()); 70 // 設置空閑的時間是30s 71 acceptor.getSessionConfig().setIdleTime(IdleStatus.BOTH_IDLE, 30); 72 try { 73 /* 74 * 4.將服務器綁定到3456端口 75 */ 76 acceptor.bind(new InetSocketAddress(3456)); 77 } catch (IOException e) { 78 e.printStackTrace(); 79 return false; 80 } 81 return true; 82 } 83 84 public static void main(String[] args) { 85 MinaServer server = new MinaServer(); 86 server.start(); 87 } 88 }
MinaServer就是按照第二步當中的流程走過來的;所以編程的時候最主要的是整體的思路,思路明白了那么編程就會變得異常效率。
- MinaServer.Java
-
-
-
- SSLContextGenerator.Java(同客戶端的代碼,看過客戶端的可以不看)
1 package com.mina.ssl; 2 3 import java.io.File; 4 import java.security.KeyStore; 5 6 import javax.net.ssl.SSLContext; 7 8 import org.apache.mina.filter.ssl.KeyStoreFactory; 9 import org.apache.mina.filter.ssl.SslContextFactory; 10 11 /** 12 * <pre> 13 * Project Name:SSLContextGenerator 14 * Package:com.example.mina.ssl 15 * FileName:SSLContextGenerator.java 16 * Purpose:SSL加密的上下文產生器 17 * Create Time: 2014-8-19 下午4:41:55 18 * Create Specification: 19 * Modified Time: 20 * Modified by: 21 * Modified Specification: 22 * Version: 1.0 23 * </pre> 24 * 25 * @author myp 26 */ 27 public class SSLContextGenerator { 28 29 /** 30 * 這個方法,通過keystore和truststore文件返回一個SSLContext對象 31 * 32 * @return 33 */ 34 public SSLContext getSslContext() { 35 SSLContext sslContext = null; 36 try { 37 /* 38 * 提供keystore的存放目錄,讀取keystore的文件內容 39 */ 40 File keyStoreFile = new File("C:/Users/Myp/keystore.jks"); 41 42 /* 43 * 提供truststore的存放目錄,讀取truststore的文件內容 44 */ 45 File trustStoreFile = new File("C:/Users/Myp/truststore.jks"); 46 47 if (keyStoreFile.exists() && trustStoreFile.exists()) { 48 final KeyStoreFactory keyStoreFactory = new KeyStoreFactory(); 49 System.out.println("Url is: " + keyStoreFile.getAbsolutePath()); 50 keyStoreFactory.setDataFile(keyStoreFile); 51 52 /* 53 * 這個是當初我們使用keytool創建keystore和truststore文件的密碼,也是上次讓你們一定要記住密碼的原因了 54 */ 55 keyStoreFactory.setPassword("123456"); 56 57 final KeyStoreFactory trustStoreFactory = new KeyStoreFactory(); 58 trustStoreFactory.setDataFile(trustStoreFile); 59 trustStoreFactory.setPassword("123456"); 60 61 final SslContextFactory sslContextFactory = new SslContextFactory(); 62 final KeyStore keyStore = keyStoreFactory.newInstance(); 63 sslContextFactory.setKeyManagerFactoryKeyStore(keyStore); 64 65 final KeyStore trustStore = trustStoreFactory.newInstance(); 66 sslContextFactory.setTrustManagerFactoryKeyStore(trustStore); 67 sslContextFactory 68 .setKeyManagerFactoryKeyStorePassword("123456"); 69 sslContext = sslContextFactory.newInstance(); 70 System.out.println("SSL provider is: " 71 + sslContext.getProvider()); 72 } else { 73 System.out 74 .println("Keystore or Truststore file does not exist"); 75 } 76 } catch (Exception ex) { 77 ex.printStackTrace(); 78 } 79 return sslContext; 80 } 81 }
如果不知道如何創建keystore和truststore文件的話,請查看我的這篇博客:http://www.cnblogs.com/getherBlog/p/3930317.html
- SSLContextGenerator.Java(同客戶端的代碼,看過客戶端的可以不看)
-
-
-
- MsgHandler.Java(同客戶端的代碼,看過客戶端的可以不看)
1 package com.mina.hanlder; 2 3 import org.apache.mina.core.service.IoHandlerAdapter; 4 import org.apache.mina.core.session.IdleStatus; 5 import org.apache.mina.core.session.IoSession; 6 import org.slf4j.Logger; 7 import org.slf4j.LoggerFactory; 8 9 /** 10 * <pre> 11 * Project Name:MsgHanler 12 * Package:com.mina.handler 13 * FileName:MsgHanler.java 14 * Purpose:I/O消息處理器,從這里我們就可以看出Mina是事件驅動的 15 * Create Time: 2014-8-19 下午4:55:55 16 * Create Specification: 17 * Modified Time: 18 * Modified by: 19 * Modified Specification: 20 * Version: 1.0 21 * </pre> 22 * 23 * @author myp 24 */ 25 public class MsgHanler extends IoHandlerAdapter { 26 private static final Logger log = LoggerFactory.getLogger(MsgHanler.class); 27 28 @Override 29 public void exceptionCaught(IoSession session, Throwable cause) 30 throws Exception { 31 // 出現異常 32 log.error("--------exception--------"); 33 super.exceptionCaught(session, cause); 34 } 35 36 @Override 37 public void messageReceived(IoSession session, Object message) 38 throws Exception { 39 // 從服務器中接收到消息后的處理 40 log.info("--------msg receive--------"); 41 log.info("Message:{}", message.toString()); 42 super.messageReceived(session, message); 43 } 44 45 @Override 46 public void messageSent(IoSession session, Object message) throws Exception {// 往服務器中發送消息 47 log.info("Message Send {}", message.toString()); 48 super.messageSent(session, message); 49 } 50 51 @Override 52 public void sessionIdle(IoSession session, IdleStatus status) 53 throws Exception { 54 // session處於空閑的時候 55 log.info("當前連接{}處於空閑狀態:{}", session.getRemoteAddress(), status); 56 } 57 58 @Override 59 public void sessionClosed(IoSession session) throws Exception { 60 // session關閉 61 log.info("Session closed {}->{}", session.getId(), 62 session.getRemoteAddress()); 63 super.sessionClosed(session); 64 } 65 }
基本上我們最主要的就是對在I/O處理器這里對收到的消息進行處理,也是編程的核心所在!
- MsgHandler.Java(同客戶端的代碼,看過客戶端的可以不看)
-
四、向服務器發送請求
1.打開CMD命令;
2.telnet 你的IP 端口號(如:telnet 192.168.191.1 3456);
3.然后就可以往服務器輸入數據,以換行結束輸入(這是因為Mina是以換行來判斷輸入是否結束的);
注意:如果遇到cmd命令提示無法識別telnet這個命令的話,你可以這樣設置:打開控制面板--》程序和功能--》打開或關閉Windows功能--》勾選然后點擊確定即可。如下圖:
4.然后我們可以看到MyEclipse的控制台輸出的信息:可以參考下圖(這里我將ssl加密注釋了):
五、注意事項(同客戶端,看過客戶端的可以不看)
-
- 關於Mina的日志過濾器誤區,不知道會不會有同學有這樣的認為,我們的log4j-1.2.17.jar就是我們的mina的日志,那么我告訴你你理解錯了,log4j是Apache的一個開源項目,通過使用Log4j,我們可以控制日志信息輸出。Mina的日志過濾器是使用了slf4j-log4j12-1.7.6.jar、slf4j-api-1.7.6.jar包;
- log4j的配置問題,如果需要使用Apache的開源項目,我們需要配置log4j.properties文件,下面是他的代碼;
1 log4j.rootCategory=INFO, stdout , R 2 3 log4j.appender.stdout=org.apache.log4j.ConsoleAppender 4 log4j.appender.stdout.layout=org.apache.log4j.PatternLayout 5 log4j.appender.stdout.layout.ConversionPattern=[QC] %p [%t] %C.%M(%L) | %m%n 6 7 log4j.appender.R=org.apache.log4j.DailyRollingFileAppender 8 log4j.appender.R.File=D\:\\Mina\\logs\\server.log 9 log4j.appender.R.layout=org.apache.log4j.PatternLayout 10 1log4j.appender.R.layout.ConversionPattern=%d-[TS] %p %t %c - %m%n 11 12 log4j.logger.com.neusoft=DEBUG 13 log4j.logger.com.opensymphony.oscache=ERROR 14 log4j.logger.net.sf.navigator=ERROR 15 log4j.logger.org.apache.commons=ERROR 16 log4j.logger.org.apache.struts=WARN 17 log4j.logger.org.displaytag=ERROR 18 log4j.logger.org.springframework=DEBUG 19 log4j.logger.com.ibatis.db=WARN 20 log4j.logger.org.apache.velocity=FATAL 21 22 log4j.logger.com.canoo.webtest=WARN 23 24 log4j.logger.org.hibernate.ps.PreparedStatementCache=WARN 25 log4j.logger.org.hibernate=DEBUG 26 log4j.logger.org.logicalcobwebs=WARN
我們可以再這里設置我們的日志輸出目錄:log4j.appender.R.File=D:\\Mina\\logs\\server.log
3. SSL加密中,如果不知道如何使用keystore生成keystore和truststore文件,可以查看這篇博客:http://www.cnblogs.com/getherBlog/p/3930317.html
4. 在添加過濾器的時候,處理的順序是按照添加過濾器的順序;
5. Mina在使用過濾器的時候,只要在需要的地方添加就可以了,不一定是服務器、客戶端都要添加的。就是說,服務器、客戶端編程的時候服務器有這個過濾器,客戶端可以有也可以沒有。
六、Mina服務器源碼下載
下載后導入到MyEclipse當中,將com.mina.example包下面的MinaServer類中的下面代碼注釋掉,然后就可以正常運行了!原因是你本地不存在keystore和 truststore文件,如果需要生成請看注意事項中第三條。
1 SslFilter sslFilter = new SslFilter( 2 new SSLContextGenerator().getSslContext()); 3 filterChain.addFirst("sslFilter", sslFilter); 4 System.out.println("SSL support is added..");
那么到這里Mina的開發就暫時告一段落了,如果還有什么需要添加的我以后會在發博客的,歡迎訂閱!我的CSDN地址:http://blog.csdn.net/u010049692/article/details/38864541