一. 开发背景

想要成为一名优秀的Android开发,你需要一份完备的 知识体系,在这里,让我们一起成长为自己所想的那样~。

我们的项目需要开发一款智能硬件。它由 Web 后台发送指令到一款桌面端应用程序,再由桌面程序来控制不同的硬件设备实现业务上的操作。从 Web 后台到桌面端是通过一个 WebSocket 长链接来进行维护,而桌面程序到各个硬件设备也是一个 TCP 长链接来维护的。

本文讲述的,其实是从桌面程序到各个硬件之间的通讯。

二. 自定义通讯协议

首先,需要设计一个通用的 TCP 网络协议。

网络协议结构如下

     +--------------+---------------+------------+---------------+-----------+----------+     | 魔数(4)       | version(1)    |序列化方式(1) | command(1)    |数据长度(4) |数据(n)    |     +--------------+---------------+------------+---------------+-----------+----------+
  • 魔数:4字节,本项目中使用 20200803(这一天编写的日子),为了防止该端口被意外调用,我们在收到报文后取前4个字节与魔数比对,如果不相同则直接拒绝并关闭连接。
  • 版本号:1字节,仅表示协议的版本号,便于协议升级时使用
  • 序列化方式:1字节,表示如何将 Java 对象转化为二进制数据,以及如何反序列化。
  • 指令:1字节,表示该消息的意图(如拍照、拍视频、心跳、App 升级等)。最多支持 2^8 种指令。
  • 数据长度:4字节,表示该字段后数据部分的长度。最多支持 2^32 位。
  • 数据:具体数据的内容。

根据上述所设计的网络协议,定义一个抽象类 Packet:

abstract class Packet {    var magic:Int? = MAGIC_NUMBER     // 魔数    var version:Byte = 1              // 版本号,当前协议的版本号为 1    abstract val serializeMethod:Byte // 序列化方式    abstract val command:Byte         // Watcher 跟 App 相互通讯的指令}

有多少个指令就需要定义多少个 Packet,下面以心跳的 Packet 为例,定义一个 HeartBeatPacket:

data class HeartBeatPacket(var msg:String = "ping",                           override val serializeMethod: Byte = Serialize.JSON,                           override val command: Byte = Commands.HEART_BEAT) : Packet() {}

HeartBeatPacket 是由 TCP 客户端发起,由 TCP 服务端接收并返回给客户端。

每个 Packet 类都包含了该 Packet 所使用的序列化方式。

/** * 序列化方式的常量列表 */interface Serialize {    companion object {        const val JSON: Byte = 0    }}

每个 Packet 也包含了其对应的 command。下面是 Commands 是指令集,支持256个指令。

/** * 指令集,支持从 -128 到 127 总共 256 个指令 */interface Commands {    companion object {        /**         * 心跳包         */        const val HEART_BEAT: Byte = 0        /**         * 登录(App 需要告诉 Watcher :cameraPosition 的位置)         */        const val LOGIN: Byte = 1        ......   }}

由于使用自定义的协议,必须要有对报文的 encode、decode,PacketManager 负责这些事情。
encode 时按照协议的结构进行组装报文,同理 decode 是其逆向的过程。

/** * 报文的管理类,对报文进行 encode、decode */object PacketManager {    fun encode(packet: Packet):ByteBuf = encode(ByteBufAllocator.DEFAULT, packet)    fun encode(alloc:ByteBufAllocator, packet: Packet) = encode(alloc.ioBuffer(), packet)    fun encode(buf: ByteBuf, packet: Packet): ByteBuf {        val serializer = SerializerFactory.getSerializer(packet.serializeMethod)        val bytes: ByteArray = serializer.serialize(packet)        //组装报文:魔数(4字节)+ 版本号(1字节)+ 序列化方式(1字节)+ 指令(1字节)+ 数据长度(4字节)+ 数据(N字节)        buf.writeInt(MAGIC_NUMBER)        buf.writeByte(packet.version.toInt())        buf.writeByte(packet.serializeMethod.toInt())        buf.writeByte(packet.command.toInt())        buf.writeInt(bytes.size)        buf.writeBytes(bytes)        return buf    }    fun decode(buf:ByteBuf): Packet {        buf.skipBytes(4) // 魔数由单独的 Handler 进行校验        buf.skipBytes(1)        val serializationMethod = buf.readByte()        val serializer = SerializerFactory.getSerializer(serializationMethod)        val command = buf.readByte()        val clazz = PacketFactory.getPacket(command)        val length = buf.readInt()  // 数据的长度        val bytes = ByteArray(length)   // 定义需要读取的字符数组        buf.readBytes(bytes)        return serializer.deserialize(clazz, bytes)    }}

三. TCP 服务端

启动 TCP 服务的方法

    fun execute() {        boss = NioEventLoopGroup()        worker = NioEventLoopGroup()        val bootstrap = ServerBootstrap()        bootstrap.group(boss, worker).channel(NioServerSocketChannel::class.java)                .option(ChannelOption.SO_BACKLOG, 100)                .childOption(ChannelOption.SO_KEEPALIVE, true)                .childOption(ChannelOption.SO_REUSEADDR, true)                .childOption(ChannelOption.TCP_NODELAY, true)                .childHandler(object : ChannelInitializer() {                    @Throws(Exception::class)                    override fun initChannel(nioSocketChannel: NioSocketChannel) {                        val pipeline = nioSocketChannel.pipeline()                        pipeline.addLast(ServerIdleHandler())                        pipeline.addLast(MagicNumValidator())                        pipeline.addLast(PacketCodecHandler)                        pipeline.addLast(HeartBeatHandler)                        pipeline.addLast(ResponseHandler)                    }                })        val future: ChannelFuture = bootstrap.bind(TCP_PORT)        future.addListener(object : ChannelFutureListener {            @Throws(Exception::class)            override fun operationComplete(channelFuture: ChannelFuture) {                if (channelFuture.isSuccess) {                    logInfo(logger, "TCP Server is starting...")                } else {                    logError(logger,channelFuture.cause(),"TCP Server failed")                }            }        })    }

其中,ServerIdleHandler: 表示 5 分钟内没有收到心跳,则断开连接。

class ServerIdleHandler : IdleStateHandler(0, 0, HERT_BEAT_TIME) {    private val logger: Logger = LoggerFactory.getLogger(ServerIdleHandler::class.java)    @Throws(Exception::class)    override fun channelIdle(ctx: ChannelHandlerContext, evt: IdleStateEvent) {        logInfo(logger) {            ctx.channel().close()            "$HERT_BEAT_TIME 秒内没有收到心跳,则断开连接"        }    }    companion object {        private const val HERT_BEAT_TIME = 300    }}

MagicNumValidator:用于 TCP 报文的魔数校验。

class MagicNumValidator : LengthFieldBasedFrameDecoder(Int.MAX_VALUE, LENGTH_FIELD_OFFSET, LENGTH_FIELD_LENGTH) {    private val logger: Logger = LoggerFactory.getLogger(this.javaClass)    @Throws(Exception::class)    override fun decode(ctx: ChannelHandlerContext, `in`: ByteBuf): Any? {        if (`in`.getInt(`in`.readerIndex()) !== MAGIC_NUMBER) { // 魔数校验不通过,则关闭连接            logInfo(logger,"魔数校验失败")            ctx.channel().close()            return null        }        return super.decode(ctx, `in`)    }    companion object {        private const val LENGTH_FIELD_OFFSET = 7        private const val LENGTH_FIELD_LENGTH = 4    }}

PacketCodecHandler: 解析报文的 Handler。

PacketCodecHandler 继承自 ByteToMessageCodec ,它是用来处理 byte-to-message 和message-to-byte,便于解码字节消息成 POJO 或编码 POJO 消息成字节。

@ChannelHandler.Sharableobject PacketCodecHandler : MessageToMessageCodec() {    override fun encode(ctx: ChannelHandlerContext, msg: Packet, list: MutableList) {        val byteBuf = ctx.channel().alloc().ioBuffer()        PacketManager.encode(byteBuf, msg)        list.add(byteBuf)    }    override fun decode(ctx: ChannelHandlerContext, msg: ByteBuf, list: MutableList) {        list.add(PacketManager.decode(msg));    }}

HeartBeatHandler:心跳的 Handler,接收 TCP 客户端发来的"ping",然后给客户端返回"pong"。

@ChannelHandler.Sharableobject HeartBeatHandler : SimpleChannelInboundHandler(){    private val logger: Logger = LoggerFactory.getLogger(this.javaClass)    override fun channelRead0(ctx: ChannelHandlerContext, msg: HeartBeatPacket) {        logInfo(logger,"收到心跳包:${GsonUtils.toJson(msg)}")        msg.msg = "pong" // 返回 pong 给到客户端        ctx.writeAndFlush(msg)    }}

ResponseHandler:通用的处理接收 TCP 客户端发来指令的 Handler,可以根据对应的指令去查询对应的 Handler 并处理其命令。

object ResponseHandler: SimpleChannelInboundHandler() {    private val logger: Logger = LoggerFactory.getLogger(this.javaClass)    private val handlerMap: ConcurrentHashMap> = ConcurrentHashMap()    init {        handlerMap[LOGIN] = LoginHandler        ......        handlerMap[ERROR] = ErrorHandler    }    override fun channelRead0(ctx: ChannelHandlerContext, msg: Packet) {        logInfo(logger,"收到客户端的指令: ${msg.command}")        val handler: SimpleChannelInboundHandler? = handlerMap[msg.command]        handler?.let {            logInfo(logger,"找到响应指令的 Handler: ${it.javaClass.simpleName}")            it.channelRead(ctx, msg)        } ?: logInfo(logger,"未找到响应指令的 Handler")    }    @Throws(Exception::class)    override fun channelInactive(ctx: ChannelHandlerContext) {        val insocket = ctx.channel().remoteAddress() as InetSocketAddress        val clientIP = insocket.address.hostAddress        val clientPort = insocket.port        logError(logger,"客户端掉线: $clientIP : $clientPort")        super.channelInactive(ctx)    }}

四. TCP 客户端

模拟一个客户端的实现

val topLevelClass = object : Any() {}.javaClass.enclosingClassval logger: Logger = LoggerFactory.getLogger(topLevelClass)fun main() {    val worker = NioEventLoopGroup()    val bootstrap = Bootstrap()    bootstrap.group(worker).channel(NioSocketChannel::class.java)            .handler(object : ChannelInitializer() {                @Throws(Exception::class)                override fun initChannel(channel: SocketChannel) {                    channel.pipeline().addLast(PacketCodecHandler)                    channel.pipeline().addLast(ClientIdleHandler())                    channel.pipeline().addLast(ClientLogin())                }            })    val future: ChannelFuture = bootstrap.connect("127.0.0.1", TCP_PORT).addListener(object : ChannelFutureListener {        @Throws(Exception::class)        override fun operationComplete(channelFuture: ChannelFuture) {            if (channelFuture.isSuccess()) {                logInfo(logger,"connect to server success!")            } else {                logger.info("failed to connect the server! ")                System.exit(0)            }        }    })    try {        future.channel().closeFuture().sync()        logInfo(logger,"与服务端断开连接!")    } catch (e: InterruptedException) {        e.printStackTrace()    }}

其中,PacketCodecHandler 跟服务端使用的解析报文的 Handler 是一样的。

ClientIdleHandler:客户端实现心跳,每隔 30 秒发送一次心跳。

class ClientIdleHandler : IdleStateHandler(0, 0, HEART_BEAT_TIME) {    private val logger = LoggerFactory.getLogger(ClientIdleHandler::class.java)    @Throws(Exception::class)    override fun channelIdle(ctx: ChannelHandlerContext, evt: IdleStateEvent?) {        logInfo(logger,"发送心跳....")        ctx.writeAndFlush(HeartBeatPacket())    }    companion object {        private const val HEART_BEAT_TIME = 30    }}

ClientLogin:登录服务端的 Handler。

@ChannelHandler.Sharableclass ClientLogin: ChannelInboundHandlerAdapter() {    private val logger: Logger = LoggerFactory.getLogger(this.javaClass)    @Throws(Exception::class)    override fun channelActive(ctx: ChannelHandlerContext) {        val packet: LoginPacket = LoginPacket()        logInfo(logger,"packet = ${GsonUtils.toJson(packet)}")        val byteBuf = PacketManager.encode(packet)        ctx.channel().writeAndFlush(byteBuf)    }}

五. 总结

这次,我开发的桌面端程序其实逻辑并不复杂,只需接收 Web 后台的指令,然后跟各个设备进行交互。

接收到 Web 端的指令后,通过 Guava 的 EventBus 将指令通过 TCP 发送给各个设备,发送时需要转化成对应的 Packet。因此,核心的模块就是这个 TCP 自定义的协议。

tcp option 结构体_基于 Kotlin 实现一个简单的 TCP 自定义协议相关推荐

  1. mysql存储过程自定义结构体_(转)MySQL存储过程/存储过程与自定义函数的区别...

    转自:http://www.cnblogs.com/caoruiy/p/4486249.html 语法: 创建存储过程: CREATE[definer = {user|current_user}] P ...

  2. python查询银行汇款_基于Python实现一个简单的银行转账操作

    前言 在进行一个应用系统的开发过程中,从上到下一般需要四个构件:客户端-业务逻辑层-数据访问层-数据库,其中数据访问层是一个底层.核心的技术.而且在实际开发中,数据库的操作也就是说数据访问层都是嵌套在 ...

  3. C 语言结构体_点运算符( . )和箭头运算符( - )的区别

    很多时候,在对结构体进行相应的编码时,时而发现是用点运算符( . ),时而是用箭头运算符( -> ) 那么这两者之间的使用有什么区别吗? 相同点 两者都是二元操作符,而且右边的操作数都是成员的名 ...

  4. C++_结构体指针_嵌套结构体_结构体做为函数参数_结构体值传递和指针传递---C++语言工作笔记026

    然后我们来看结构体指针. 可以看到我们先去定义一个结构体 然后我们在main函数中,去声明一个结构体 s 然后我们定义一个指针 int *p = &s; 指向这个结构体变量. 这里要注意

  5. 基于PHP实现一个简单的在线聊天功能(轮询ajax )

    基于PHP实现一个简单的在线聊天功能(轮询ajax ) 一.总结 1.用的轮询ajax 二.基于PHP实现一个简单的在线聊天功能 一直很想试着做一做这个有意思的功能,感觉复杂的不是数据交互和表结构,麻 ...

  6. 基于 Roslyn 实现一个简单的条件解析引擎

    基于 Roslyn 实现一个简单的条件解析引擎 Intro 最近在做一个勋章的服务,我们想定义一些勋章的获取条件,满足条件之后就给用户颁发一个勋章,定义条件的时候会定义需要哪些参数,参数的类型,获取勋 ...

  7. 笨办法学C 练习45:一个简单的TCP/IP客户端

    练习45:一个简单的TCP/IP客户端 原文:Exercise 45: A Simple TCP/IP Client 译者:飞龙 我打算使用RingBuffer来创建一个非常简单的小型网络测试工具,叫 ...

  8. 基于ForkJoin构建一个简单易用的并发组件

    2019独角兽企业重金招聘Python工程师标准>>> 基于ForkJoin构建一个简单易用的并发组件 在实际的业务开发中,需要用到并发编程的知识,实际使用线程池来异步执行任务的场景 ...

  9. 我的Serverless实战—基于Serverless搭建一个简单的WordPress个人博客图文详解-JJZ

    文正在参与 "100%有奖 | 我的Serverless 实战"征稿活动 活动链接:https://marketing.csdn.net/p/15940c87f66c68188cf ...

最新文章

  1. CentOS 7 yum安装Zabbix
  2. 小程序webview不全屏_有赞微信商城和有赞微信小程序什么不一样
  3. JS JavaScript模块化(ES Module/CommonJS/AMD/CMD)
  4. linux 多线程 写日志,rsyslog多线程远程日志记录介绍(lamp+rsyslog)
  5. c打印无符号整数_C语言基础知识:printf的输出格式,C/C++语言编程讲解
  6. 练习:查找指定目录(包括子目录)下的视频(格式为.mp4,.rmvb,.avi),并将目录存放在一个文件中...
  7. 520套电商行业响应式html5模板b2c商城购物网站模板HTML5化妆品电商网站模板IT类电子商务商城购物企业网站模板html5网页静态模板Bootstrap扁平化网站源码
  8. pyserial串口学习
  9. 模式分解无损连接判断——数据库考试复习
  10. HTML5基本标签使用header,nav和footer
  11. 新手小白零基础,该怎样学习编程呢?
  12. scratch成绩查询 电子学会图形化编程scratch等级考试四级真题和答案解析2022年6月
  13. php页面跳底部,监控页面滑到底部加载事件的jq
  14. python动态爬虫_Python动态网页爬虫技术
  15. zigbee CC2530 系列教程 8 AD采集内部温度实验
  16. 跨平台调用之一——java调用so库
  17. 7段并行数码管显示实验C语言,七段数码管显示十进制数字 (15)
  18. 除了“带地球去流浪”,还有哪些脑洞大开的太空计划?
  19. sql Sever 2012安装失败解决办法!
  20. JavaScript进阶(四)

热门文章

  1. MySQL之无限级分类表设计
  2. Mybatis的selectKey使用
  3. MySQL带BETWEEN AND关键字的查询
  4. python数据对比找不同,不同模式间的数据比较技术
  5. bat等大公司常考java多线程面试题
  6. java基础提升篇:深入浅出Java多线程
  7. spring事务(Transaction)的七种事务传播行为及五种隔离级别
  8. 网络编程4之UDP协议
  9. 位运算的那些奇技淫巧 | 掌(装)握(逼)必备,妙解两道算法题
  10. UML图系列——用例图