【Andorid开发框架学习】之Mina开发之客户端开发

2024-05-04 16:08

本文主要是介绍【Andorid开发框架学习】之Mina开发之客户端开发,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

之前我们讲到了Mina的基本知识点。如果还有不懂得同学可以看一下我的博客:我是传送门。今天我着重来讲一下基于Mina的客户端的开发(代码均在最后链接地址中,欢迎下载)。

 

   一、首先看一下,我的客户端的代码图片:

       客户端代码我是在Eclipse下写的。

 

   二、客户端的整体思路:

    • 首先,产生一个socket连接对象,用于连接到服务器;
    • 然后,对这个连接添加我们的I/O过滤器(SSL加密、日志过滤器、编码过滤器等,这里注意,如果添加SSL过滤器,那么一定要第一个添加,否则无法对数据加密);
    • 接下来,为连接设置I/O处理器,顾名思义就是处理接收到的消息(这里我们只能设置一个处理器,如果有设置多个,那么默认进入到最后一个I/O处理器中进行处理);
    • 最后,连接到通过IP和端口号连接到服务器;保存连接的获取到的session,如果需要发送消息,我们就可以对session进行操作。

 

   三、正式编码

      这里我展示几个比较重要的类来详细说明一下:

      • MinaClient.Java
        package com.example.mina.server;import java.net.InetSocketAddress;import org.apache.mina.core.filterchain.DefaultIoFilterChainBuilder;
        import org.apache.mina.core.future.CloseFuture;
        import org.apache.mina.core.future.ConnectFuture;
        import org.apache.mina.core.session.IoSession;
        import org.apache.mina.filter.codec.ProtocolCodecFilter;
        import org.apache.mina.filter.logging.LoggingFilter;
        import org.apache.mina.filter.ssl.SslFilter;
        import org.apache.mina.transport.socket.SocketConnector;
        import org.apache.mina.transport.socket.nio.NioSocketConnector;import com.example.mina.charset.CharsetFactory;
        import com.example.mina.hanlder.MsgHanler;
        import com.example.mina.ssl.SSLContextGenerator;/*** <pre>* Project Name:MinaClient* Package:com.example.mina.server* FileName:MinaClient.java* Purpose:客户端* Create Time: 2014-8-19 下午4:36:55* Create Specification:* Modified Time:* Modified by:* Modified Specification:* Version: 1.0* </pre>* * @author myp*/
        public class MinaClient {private SocketConnector connector;private ConnectFuture future;private IoSession session;public boolean connect() {/** 1.创建一个socket连接,连接到服务器*/connector = new NioSocketConnector();/** 获取过滤器链,用于添加过滤器*/DefaultIoFilterChainBuilder chain = connector.getFilterChain();/** 2.为连接添加过滤器,SSL、日志、编码过滤器*/// SSLContextGenerator是我们自己写的一个SSL上下文产生器,稍后会讲到SslFilter sslFilter = new SslFilter(new SSLContextGenerator().getSslContext());// 设置为客户端模式sslFilter.setUseClientMode(true);// a.ssl过滤器,这个一定要第一个添加,否则数据不会进行加密chain.addFirst("sslFilter", sslFilter);// b.添加日志过滤器chain.addLast("logger", new LoggingFilter());// c.添加字符的编码过滤器chain.addLast("codec", new ProtocolCodecFilter(new CharsetFactory()));/** 3.设置消息处理器,用于处理接收到的消息*/connector.setHandler(new MsgHanler());/** 4.根据IP和端口号连接到服务器*/future = connector.connect(new InetSocketAddress("192.168.1.12", 3456));// 等待连接创建完成future.awaitUninterruptibly();/** 5.获取session对象,通过session可以向服务器发送消息;*/session = future.getSession();session.getConfig().setUseReadOperation(true);return future.isConnected();}/*** 往服务器发送消息* * @param message*/public void sendMsg2Server(String message) {session.write(message);}/*** 关闭与服务器的连接* * @return*/public boolean close() {CloseFuture future = session.getCloseFuture();future.awaitUninterruptibly(1000);connector.dispose();return true;}
        }

        MinaClient就是按照第二步当中的流程走过来的;所以编程的时候最主要的是整体的思路,思路明白了那么编程就会变得异常效率。

 

    • SSLContextGenerator.Java
      package com.example.mina.ssl;import java.io.File;
      import java.security.KeyStore;import javax.net.ssl.SSLContext;import org.apache.mina.filter.ssl.KeyStoreFactory;
      import org.apache.mina.filter.ssl.SslContextFactory;/*** <pre>* Project Name:SSLContextGenerator* Package:com.example.mina.ssl* FileName:SSLContextGenerator.java* Purpose:客户端* Create Time: 2014-8-19 下午4:41:55* Create Specification:* Modified Time:* Modified by:* Modified Specification:* Version: 1.0* </pre>* * @author myp*/
      public class SSLContextGenerator {/*** 这个方法,通过keystore和truststore文件返回一个SSLContext对象* * @return*/public SSLContext getSslContext() {SSLContext sslContext = null;try {/** 提供keystore的存放目录,读取keystore的文件内容*/File keyStoreFile = new File("C:/Users/Administrator/keystore.jks");/** 提供truststore的存放目录,读取truststore的文件内容*/File trustStoreFile = new File("C:/Users/Administrator/truststore.jks");if (keyStoreFile.exists() && trustStoreFile.exists()) {final KeyStoreFactory keyStoreFactory = new KeyStoreFactory();System.out.println("Url is: " + keyStoreFile.getAbsolutePath());keyStoreFactory.setDataFile(keyStoreFile);/** 这个是当初我们使用keytool创建keystore和truststore文件的密码,也是上次让你们一定要记住密码的原因了*/keyStoreFactory.setPassword("123456");final KeyStoreFactory trustStoreFactory = new KeyStoreFactory();trustStoreFactory.setDataFile(trustStoreFile);trustStoreFactory.setPassword("123456");final SslContextFactory sslContextFactory = new SslContextFactory();final KeyStore keyStore = keyStoreFactory.newInstance();sslContextFactory.setKeyManagerFactoryKeyStore(keyStore);final KeyStore trustStore = trustStoreFactory.newInstance();sslContextFactory.setTrustManagerFactoryKeyStore(trustStore);sslContextFactory.setKeyManagerFactoryKeyStorePassword("123456");sslContext = sslContextFactory.newInstance();System.out.println("SSL provider is: "+ sslContext.getProvider());} else {System.out.println("Keystore or Truststore file does not exist");}} catch (Exception ex) {ex.printStackTrace();}return sslContext;}
      }

      如果不知道如何创建keystore和truststore文件的话,请查看我的这篇博客:http://blog.csdn.net/u010049692/article/details/38686659

 

    • MsgHandler.Java
      package com.example.mina.hanlder;import org.apache.mina.core.service.IoHandlerAdapter;
      import org.apache.mina.core.session.IoSession;
      import org.slf4j.Logger;
      import org.slf4j.LoggerFactory;/*** <pre>* Project Name:MsgHanler* Package:com.example.mina.handler* FileName:MsgHanler.java* Purpose:I/O消息处理器,从这里我们就可以看出Mina是事件驱动的* Create Time: 2014-8-19 下午4:39:55* Create Specification:* Modified Time:* Modified by:* Modified Specification:* Version: 1.0* </pre>* * @author myp*/
      public class MsgHanler extends IoHandlerAdapter {private static final Logger log = LoggerFactory.getLogger(MsgHanler.class);@Overridepublic void exceptionCaught(IoSession session, Throwable cause)throws Exception {// 出现异常log.error("--------exception--------");super.exceptionCaught(session, cause);}@Overridepublic void messageReceived(IoSession session, Object message)throws Exception {// 从服务器中接收到消息后的处理log.info("--------msg receive--------");log.info("Message:{}" + message.toString());super.messageReceived(session, message);}@Overridepublic void messageSent(IoSession session, Object message) throws Exception {// 往服务器中发送消息log.info("--------msg sent--------");super.messageSent(session, message);}@Overridepublic void sessionCreated(IoSession session) throws Exception {// 当session被创建的时候调用log.info("--------session create--------");super.sessionCreated(session);}
      }

      基本上我们最主要的就是对在I/O处理器这里对收到的消息进行处理,也是编程的核心所在!

       

   四、注意事项

      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文件,下面是他的代码;
        log4j.rootCategory=INFO, stdout , R   log4j.appender.stdout=org.apache.log4j.ConsoleAppender   
        log4j.appender.stdout.layout=org.apache.log4j.PatternLayout   
        log4j.appender.stdout.layout.ConversionPattern=[QC] %p [%t] %C.%M(%L) | %m%n   log4j.appender.R=org.apache.log4j.DailyRollingFileAppender  
        log4j.appender.R.File=D:\\Mina\\logs\\client.log   
        log4j.appender.R.layout=org.apache.log4j.PatternLayout   
        1log4j.appender.R.layout.ConversionPattern=%d-[TS] %p %t %c - %m%n   log4j.logger.com.neusoft=DEBUG   
        log4j.logger.com.opensymphony.oscache=ERROR   
        log4j.logger.net.sf.navigator=ERROR   
        log4j.logger.org.apache.commons=ERROR   
        log4j.logger.org.apache.struts=WARN   
        log4j.logger.org.displaytag=ERROR   
        log4j.logger.org.springframework=DEBUG   
        log4j.logger.com.ibatis.db=WARN   
        log4j.logger.org.apache.velocity=FATAL   log4j.logger.com.canoo.webtest=WARN   log4j.logger.org.hibernate.ps.PreparedStatementCache=WARN   
        log4j.logger.org.hibernate=DEBUG   
        log4j.logger.org.logicalcobwebs=WARN  log4j.rootCategory=INFO, stdout , Rlog4j.appender.stdout=org.apache.log4j.ConsoleAppender
        log4j.appender.stdout.layout=org.apache.log4j.PatternLayout
        log4j.appender.stdout.layout.ConversionPattern=[QC] %p [%t] %C.%M(%L) | %m%nlog4j.appender.R=org.apache.log4j.DailyRollingFileAppender
        log4j.appender.R.File=D:\\Tomcat 5.5\\logs\\qc.log
        log4j.appender.R.layout=org.apache.log4j.PatternLayout
        1log4j.appender.R.layout.ConversionPattern=%d-[TS] %p %t %c - %m%nlog4j.logger.com.neusoft=DEBUG
        log4j.logger.com.opensymphony.oscache=ERROR
        log4j.logger.net.sf.navigator=ERROR
        log4j.logger.org.apache.commons=ERROR
        log4j.logger.org.apache.struts=WARN
        log4j.logger.org.displaytag=ERROR
        log4j.logger.org.springframework=DEBUG
        log4j.logger.com.ibatis.db=WARN
        log4j.logger.org.apache.velocity=FATALlog4j.logger.com.canoo.webtest=WARNlog4j.logger.org.hibernate.ps.PreparedStatementCache=WARN
        log4j.logger.org.hibernate=DEBUG
        log4j.logger.org.logicalcobwebs=WARN

        我们可以再这里设置我们的日志输出目录:log4j.appender.R.File=D:\\Mina\\logs\\client.log   

         3. SSL加密中,如果不知道如何使用keystore生成keystore和truststore文件,可以查看这篇博客:http://blog.csdn.net/u010049692/article/details/38686659

         4. 在添加过滤器的时候,处理的顺序是按照添加过滤器的顺序

         5. Mina在使用过滤器的时候,只要在需要的地方添加就可以了,不一定是服务器、客户端都要添加的。就是说,服务器、客户端编程的时候服务器有这个过滤器,客户端可以有也可以没有。

        

  

   五、Mina客户端源码下载

        点我下载

        下载后导入到Eclipse当中,将com.example.mina.server包下面的MinaClient类中的下面代码注释掉,然后就可以正常运行了!原因是你本地不存在keystore和truststore文件,如果需要生成请看注意事项中第三条。

    SslFilter sslFilter = new SslFilter(new SSLContextGenerator().getSslContext());sslFilter.setUseClientMode(true);chain.addFirst("sslFilter", sslFilter);

        

     

   下一篇应该是Mina的服务器的开发,欢迎订阅!

我的博客园地址:http://www.cnblogs.com/getherBlog/p/3937196.html

这篇关于【Andorid开发框架学习】之Mina开发之客户端开发的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/959585

相关文章

HarmonyOS学习(七)——UI(五)常用布局总结

自适应布局 1.1、线性布局(LinearLayout) 通过线性容器Row和Column实现线性布局。Column容器内的子组件按照垂直方向排列,Row组件中的子组件按照水平方向排列。 属性说明space通过space参数设置主轴上子组件的间距,达到各子组件在排列上的等间距效果alignItems设置子组件在交叉轴上的对齐方式,且在各类尺寸屏幕上表现一致,其中交叉轴为垂直时,取值为Vert

Ilya-AI分享的他在OpenAI学习到的15个提示工程技巧

Ilya(不是本人,claude AI)在社交媒体上分享了他在OpenAI学习到的15个Prompt撰写技巧。 以下是详细的内容: 提示精确化:在编写提示时,力求表达清晰准确。清楚地阐述任务需求和概念定义至关重要。例:不用"分析文本",而用"判断这段话的情感倾向:积极、消极还是中性"。 快速迭代:善于快速连续调整提示。熟练的提示工程师能够灵活地进行多轮优化。例:从"总结文章"到"用

这15个Vue指令,让你的项目开发爽到爆

1. V-Hotkey 仓库地址: github.com/Dafrok/v-ho… Demo: 戳这里 https://dafrok.github.io/v-hotkey 安装: npm install --save v-hotkey 这个指令可以给组件绑定一个或多个快捷键。你想要通过按下 Escape 键后隐藏某个组件,按住 Control 和回车键再显示它吗?小菜一碟: <template

Hadoop企业开发案例调优场景

需求 (1)需求:从1G数据中,统计每个单词出现次数。服务器3台,每台配置4G内存,4核CPU,4线程。 (2)需求分析: 1G / 128m = 8个MapTask;1个ReduceTask;1个mrAppMaster 平均每个节点运行10个 / 3台 ≈ 3个任务(4    3    3) HDFS参数调优 (1)修改:hadoop-env.sh export HDFS_NAMENOD

【前端学习】AntV G6-08 深入图形与图形分组、自定义节点、节点动画(下)

【课程链接】 AntV G6:深入图形与图形分组、自定义节点、节点动画(下)_哔哩哔哩_bilibili 本章十吾老师讲解了一个复杂的自定义节点中,应该怎样去计算和绘制图形,如何给一个图形制作不间断的动画,以及在鼠标事件之后产生动画。(有点难,需要好好理解) <!DOCTYPE html><html><head><meta charset="UTF-8"><title>06

学习hash总结

2014/1/29/   最近刚开始学hash,名字很陌生,但是hash的思想却很熟悉,以前早就做过此类的题,但是不知道这就是hash思想而已,说白了hash就是一个映射,往往灵活利用数组的下标来实现算法,hash的作用:1、判重;2、统计次数;

嵌入式QT开发:构建高效智能的嵌入式系统

摘要: 本文深入探讨了嵌入式 QT 相关的各个方面。从 QT 框架的基础架构和核心概念出发,详细阐述了其在嵌入式环境中的优势与特点。文中分析了嵌入式 QT 的开发环境搭建过程,包括交叉编译工具链的配置等关键步骤。进一步探讨了嵌入式 QT 的界面设计与开发,涵盖了从基本控件的使用到复杂界面布局的构建。同时也深入研究了信号与槽机制在嵌入式系统中的应用,以及嵌入式 QT 与硬件设备的交互,包括输入输出设

OpenHarmony鸿蒙开发( Beta5.0)无感配网详解

1、简介 无感配网是指在设备联网过程中无需输入热点相关账号信息,即可快速实现设备配网,是一种兼顾高效性、可靠性和安全性的配网方式。 2、配网原理 2.1 通信原理 手机和智能设备之间的信息传递,利用特有的NAN协议实现。利用手机和智能设备之间的WiFi 感知订阅、发布能力,实现了数字管家应用和设备之间的发现。在完成设备间的认证和响应后,即可发送相关配网数据。同时还支持与常规Sof

活用c4d官方开发文档查询代码

当你问AI助手比如豆包,如何用python禁止掉xpresso标签时候,它会提示到 这时候要用到两个东西。https://developers.maxon.net/论坛搜索和开发文档 比如这里我就在官方找到正确的id描述 然后我就把参数标签换过来

零基础学习Redis(10) -- zset类型命令使用

zset是有序集合,内部除了存储元素外,还会存储一个score,存储在zset中的元素会按照score的大小升序排列,不同元素的score可以重复,score相同的元素会按照元素的字典序排列。 1. zset常用命令 1.1 zadd  zadd key [NX | XX] [GT | LT]   [CH] [INCR] score member [score member ...]