【Andorid開發框架學習】之Mina開發之服務器開發


    

    下午那篇博客我們講到了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 }
        View Code

        MinaServer就是按照第二步當中的流程走過來的;所以編程的時候最主要的是整體的思路,思路明白了那么編程就會變得異常效率。

 

      • 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 }
        View Code

        如果不知道如何創建keystore和truststore文件的話,請查看我的這篇博客:http://www.cnblogs.com/getherBlog/p/3930317.html

 

      • 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 }
        View Code

        基本上我們最主要的就是對在I/O處理器這里對收到的消息進行處理,也是編程的核心所在!

 

 

   四、向服務器發送請求

        1.打開CMD命令;

 

        2.telnet 你的IP 端口號(如:telnet 192.168.191.1 3456);

 

        3.然后就可以往服務器輸入數據,以換行結束輸入(這是因為Mina是以換行來判斷輸入是否結束的);

          注意:如果遇到cmd命令提示無法識別telnet這個命令的話,你可以這樣設置:打開控制面板--》程序和功能--》打開或關閉Windows功能--》勾選然后點擊確定即可。如下圖:  

               

 

        4.然后我們可以看到MyEclipse的控制台輸出的信息:可以參考下圖(這里我將ssl加密注釋了):

                         

 

   五、注意事項(同客戶端,看過客戶端的可以不看)

    1. 關於Mina的日志過濾器誤區,不知道會不會有同學有這樣的認為,我們的log4j-1.2.17.jar就是我們的mina的日志,那么我告訴你你理解錯了,log4j是Apache的一個開源項目,通過使用Log4j,我們可以控制日志信息輸出。Mina的日志過濾器是使用了slf4j-log4j12-1.7.6.jar、slf4j-api-1.7.6.jar包;                                  
    2. 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  
      View Code

      我們可以再這里設置我們的日志輸出目錄: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

 


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM