精品欧美一区二区三区在线观看 _久久久久国色av免费观看性色_国产精品久久在线观看_亚洲第一综合网站_91精品又粗又猛又爽_小泽玛利亚一区二区免费_91亚洲精品国偷拍自产在线观看 _久久精品视频在线播放_美女精品久久久_欧美日韩国产成人在线

基于 Kotlin 實現(xiàn)一個簡單的 TCP 自定義協(xié)議

開發(fā) 前端 通信技術(shù)
想要成為一名優(yōu)秀的Android開發(fā),你需要一份完備的 知識體系,在這里,讓我們一起成長為自己所想的那樣~。

一. 開發(fā)背景

想要成為一名優(yōu)秀的Android開發(fā),你需要一份完備的 知識體系,在這里,讓我們一起成長為自己所想的那樣~。

我們的項目需要開發(fā)一款智能硬件。它由 Web 后臺發(fā)送指令到一款桌面端應(yīng)用程序,再由桌面程序來控制不同的硬件設(shè)備實現(xiàn)業(yè)務(wù)上的操作。從 Web 后臺到桌面端是通過一個 WebSocket 長鏈接來進行維護,而桌面程序到各個硬件設(shè)備也是一個 TCP 長鏈接來維護的。

[[342699]]

本文講述的,其實是從桌面程序到各個硬件之間的通訊。

二. 自定義通訊協(xié)議

首先,需要設(shè)計一個通用的 TCP 網(wǎng)絡(luò)協(xié)議。

網(wǎng)絡(luò)協(xié)議結(jié)構(gòu)如下

  1. +--------------+---------------+------------+---------------+-----------+----------+ 
  2.      | 魔數(shù)(4)       | version(1)    |序列化方式(1) | command(1)    |數(shù)據(jù)長度(4) |數(shù)據(jù)(n)    | 
  3.      +--------------+---------------+------------+---------------+-----------+----------+ 
  • 魔數(shù):4字節(jié),本項目中使用 20200803(這一天編寫的日子),為了防止該端口被意外調(diào)用,我們在收到報文后取前4個字節(jié)與魔數(shù)比對,如果不相同則直接拒絕并關(guān)閉連接。
  • 版本號:1字節(jié),僅表示協(xié)議的版本號,便于協(xié)議升級時使用
  • 序列化方式:1字節(jié),表示如何將 Java 對象轉(zhuǎn)化為二進制數(shù)據(jù),以及如何反序列化。
  • 指令:1字節(jié),表示該消息的意圖(如拍照、拍視頻、心跳、App 升級等)。最多支持 2^8 種指令。
  • 數(shù)據(jù)長度:4字節(jié),表示該字段后數(shù)據(jù)部分的長度。最多支持 2^32 位。
  • 數(shù)據(jù):具體數(shù)據(jù)的內(nèi)容。

根據(jù)上述所設(shè)計的網(wǎng)絡(luò)協(xié)議,定義一個抽象類 Packet:

  1. abstract class Packet { 
  2.     var magic:Int? = MAGIC_NUMBER     // 魔數(shù) 
  3.     var version:Byte = 1              // 版本號,當(dāng)前協(xié)議的版本號為 1 
  4.     abstract val serializeMethod:Byte // 序列化方式 
  5.     abstract val command:Byte         // Watcher 跟 App 相互通訊的指令 

有多少個指令就需要定義多少個 Packet,下面以心跳的 Packet 為例,定義一個 HeartBeatPacket:

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

HeartBeatPacket 是由 TCP 客戶端發(fā)起,由 TCP 服務(wù)端接收并返回給客戶端。

每個 Packet 類都包含了該 Packet 所使用的序列化方式。

  1. /** 
  2.  * 序列化方式的常量列表 
  3.  */ 
  4. interface Serialize { 
  5.     companion object { 
  6.         const val JSON: Byte = 0 
  7.     }} 

每個 Packet 也包含了其對應(yīng)的 command。下面是 Commands 是指令集,支持256個指令。

  1. /** 
  2.  * 指令集,支持從 -128 到 127 總共 256 個指令 
  3.  */ 
  4. interface Commands { 
  5.     companion object { 
  6.         /** 
  7.          * 心跳包 
  8.          */ 
  9.         const val HEART_BEAT: Byte = 0 
  10.         /** 
  11.          * 登錄(App 需要告訴 Watcher :cameraPosition 的位置) 
  12.          */ 
  13.         const val LOGIN: Byte = 1 
  14.         ......   }} 

由于使用自定義的協(xié)議,必須要有對報文的 encode、decode,PacketManager 負責(zé)這些事情。

encode 時按照協(xié)議的結(jié)構(gòu)進行組裝報文,同理 decode 是其逆向的過程。

  1. /** 
  2.  * 報文的管理類,對報文進行 encode、decode 
  3.  */ 
  4. object PacketManager { 
  5.     fun encode(packet: Packet):ByteBuf = encode(ByteBufAllocator.DEFAULT, packet) 
  6.     fun encode(alloc:ByteBufAllocator, packet: Packet) = encode(alloc.ioBuffer(), packet) 
  7.     fun encode(buf: ByteBuf, packet: Packet): ByteBuf { 
  8.         val serializer = SerializerFactory.getSerializer(packet.serializeMethod) 
  9.         val bytes: ByteArray = serializer.serialize(packet) 
  10.         //組裝報文:魔數(shù)(4字節(jié))+ 版本號(1字節(jié))+ 序列化方式(1字節(jié))+ 指令(1字節(jié))+ 數(shù)據(jù)長度(4字節(jié))+ 數(shù)據(jù)(N字節(jié)) 
  11.         buf.writeInt(MAGIC_NUMBER) 
  12.         buf.writeByte(packet.version.toInt()) 
  13.         buf.writeByte(packet.serializeMethod.toInt()) 
  14.         buf.writeByte(packet.command.toInt()) 
  15.         buf.writeInt(bytes.size
  16.         buf.writeBytes(bytes) 
  17.         return buf 
  18.     } 
  19.     fun decode(buf:ByteBuf): Packet { 
  20.         buf.skipBytes(4) // 魔數(shù)由單獨的 Handler 進行校驗 
  21.         buf.skipBytes(1) 
  22.         val serializationMethod = buf.readByte() 
  23.         val serializer = SerializerFactory.getSerializer(serializationMethod) 
  24.         val command = buf.readByte() 
  25.         val clazz = PacketFactory.getPacket(command) 
  26.         val length = buf.readInt()  // 數(shù)據(jù)的長度 
  27.         val bytes = ByteArray(length)   // 定義需要讀取的字符數(shù)組 
  28.         buf.readBytes(bytes) 
  29.         return serializer.deserialize(clazz, bytes) 
  30.     } 

三. TCP 服務(wù)端

啟動 TCP 服務(wù)的方法

  1. fun execute() { 
  2.     boss = NioEventLoopGroup()        worker = NioEventLoopGroup()        val bootstrap = ServerBootstrap() 
  3.     bootstrap.group(boss, worker).channel(NioServerSocketChannel::class.java) 
  4.             .option(ChannelOption.SO_BACKLOG, 100) 
  5.             .childOption(ChannelOption.SO_KEEPALIVE, true
  6.             .childOption(ChannelOption.SO_REUSEADDR, true
  7.             .childOption(ChannelOption.TCP_NODELAY, true
  8.             .childHandler(object : ChannelInitializer<NioSocketChannel>() { 
  9.                 @Throws(Exception::class) 
  10.                 override fun initChannel(nioSocketChannel: NioSocketChannel) { 
  11.                     val pipeline = nioSocketChannel.pipeline() 
  12.                     pipeline.addLast(ServerIdleHandler())                        pipeline.addLast(MagicNumValidator())                        pipeline.addLast(PacketCodecHandler)                        pipeline.addLast(HeartBeatHandler)                        pipeline.addLast(ResponseHandler)                    }                })        val future: ChannelFuture = bootstrap.bind(TCP_PORT) 
  13.     future.addListener(object : ChannelFutureListener { 
  14.         @Throws(Exception::class) 
  15.         override fun operationComplete(channelFuture: ChannelFuture) { 
  16.             if (channelFuture.isSuccess) { 
  17.                 logInfo(logger, "TCP Server is starting..."
  18.             } else { 
  19.                 logError(logger,channelFuture.cause(),"TCP Server failed"
  20.             }            }        })    } 

其中,ServerIdleHandler: 表示 5 分鐘內(nèi)沒有收到心跳,則斷開連接。

  1. class ServerIdleHandler : IdleStateHandler(0, 0, HERT_BEAT_TIME) { 
  2.     private val logger: Logger = LoggerFactory.getLogger(ServerIdleHandler::class.java) 
  3.     @Throws(Exception::class) 
  4.     override fun channelIdle(ctx: ChannelHandlerContext, evt: IdleStateEvent) { 
  5.         logInfo(logger) {            ctx.channel().close()            "$HERT_BEAT_TIME 秒內(nèi)沒有收到心跳,則斷開連接" 
  6.         }    }    companion object { 
  7.         private const val HERT_BEAT_TIME = 300 
  8.     }} 

MagicNumValidator:用于 TCP 報文的魔數(shù)校驗。

  1. class MagicNumValidator : LengthFieldBasedFrameDecoder(Int.MAX_VALUE, LENGTH_FIELD_OFFSET, LENGTH_FIELD_LENGTH) { 
  2.     private val logger: Logger = LoggerFactory.getLogger(this.javaClass) 
  3.     @Throws(Exception::class) 
  4.     override fun decode(ctx: ChannelHandlerContext, `in`: ByteBuf): Any? { 
  5.         if (`in`.getInt(`in`.readerIndex()) !== MAGIC_NUMBER) { // 魔數(shù)校驗不通過,則關(guān)閉連接 
  6.             logInfo(logger,"魔數(shù)校驗失敗"
  7.             ctx.channel().close() 
  8.             return null 
  9.         } 
  10.         return super.decode(ctx, `in`) 
  11.     } 
  12.     companion object { 
  13.         private const val LENGTH_FIELD_OFFSET = 7 
  14.         private const val LENGTH_FIELD_LENGTH = 4 
  15.     } 

PacketCodecHandler: 解析報文的 Handler。

PacketCodecHandler 繼承自 ByteToMessageCodec ,它是用來處理 byte-to-message 和message-to-byte,便于解碼字節(jié)消息成 POJO 或編碼 POJO 消息成字節(jié)。

  1. @ChannelHandler.Sharable 
  2. object PacketCodecHandler : MessageToMessageCodec<ByteBuf, Packet>() {    override fun encode(ctx: ChannelHandlerContext, msg: Packet, list: MutableList<Any>) { 
  3.         val byteBuf = ctx.channel().alloc().ioBuffer() 
  4.         PacketManager.encode(byteBuf, msg)        list.add(byteBuf)    }    override fun decode(ctx: ChannelHandlerContext, msg: ByteBuf, list: MutableList<Any>) { 
  5.         list.add(PacketManager.decode(msg));    }} 

HeartBeatHandler:心跳的 Handler,接收 TCP 客戶端發(fā)來的"ping",然后給客戶端返回"pong"。

  1. @ChannelHandler.Sharable 
  2. object HeartBeatHandler : SimpleChannelInboundHandler<HeartBeatPacket>(){    private val logger: Logger = LoggerFactory.getLogger(this.javaClass) 
  3.     override fun channelRead0(ctx: ChannelHandlerContext, msg: HeartBeatPacket) { 
  4.         logInfo(logger,"收到心跳包:${GsonUtils.toJson(msg)}"
  5.         msg.msg = "pong" // 返回 pong 給到客戶端 
  6.         ctx.writeAndFlush(msg) 
  7.     } 

ResponseHandler:通用的處理接收 TCP 客戶端發(fā)來指令的 Handler,可以根據(jù)對應(yīng)的指令去查詢對應(yīng)的 Handler 并處理其命令。

  1. object ResponseHandler: SimpleChannelInboundHandler<Packet>() { 
  2.     private val logger: Logger = LoggerFactory.getLogger(this.javaClass) 
  3.     private val handlerMap: ConcurrentHashMap<Byte, SimpleChannelInboundHandler<out Packet>> = ConcurrentHashMap() 
  4.     init { 
  5.         handlerMap[LOGIN] = LoginHandler        ......        handlerMap[ERROR] = ErrorHandler    }    override fun channelRead0(ctx: ChannelHandlerContext, msg: Packet) { 
  6.         logInfo(logger,"收到客戶端的指令: ${msg.command}"
  7.         val handler: SimpleChannelInboundHandler<out Packet>? = handlerMap[msg.command] 
  8.         handler?.let {            logInfo(logger,"找到響應(yīng)指令的 Handler: ${it.javaClass.simpleName}"
  9.             it.channelRead(ctx, msg)        } ?: logInfo(logger,"未找到響應(yīng)指令的 Handler"
  10.     }    @Throws(Exception::class) 
  11.     override fun channelInactive(ctx: ChannelHandlerContext) { 
  12.         val insocket = ctx.channel().remoteAddress() as InetSocketAddress 
  13.         val clientIP = insocket.address.hostAddress 
  14.         val clientPort = insocket.port 
  15.         logError(logger,"客戶端掉線: $clientIP : $clientPort"
  16.         super.channelInactive(ctx) 
  17.     }} 

四. TCP 客戶端

模擬一個客戶端的實現(xiàn)

  1. val topLevelClass = object : Any() {}.javaClass.enclosingClass 
  2. val logger: Logger = LoggerFactory.getLogger(topLevelClass)fun main() { 
  3.     val worker = NioEventLoopGroup() 
  4.     val bootstrap = Bootstrap() 
  5.     bootstrap.group(worker).channel(NioSocketChannel::class.java) 
  6.             .handler(object : ChannelInitializer<SocketChannel>() { 
  7.                 @Throws(Exception::class) 
  8.                 override fun initChannel(channel: SocketChannel) { 
  9.                     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 { 
  10.         @Throws(Exception::class) 
  11.         override fun operationComplete(channelFuture: ChannelFuture) { 
  12.             if (channelFuture.isSuccess()) { 
  13.                 logInfo(logger,"connect to server success!"
  14.             } else { 
  15.                 logger.info("failed to connect the server! "
  16.                 System.exit(0) 
  17.             }        }    })    try { 
  18.         future.channel().closeFuture().sync()        logInfo(logger,"與服務(wù)端斷開連接!"
  19.     } catch (e: InterruptedException) { 
  20.         e.printStackTrace()    }} 

其中,PacketCodecHandler 跟服務(wù)端使用的解析報文的 Handler 是一樣的。

ClientIdleHandler:客戶端實現(xiàn)心跳,每隔 30 秒發(fā)送一次心跳。

  1. class ClientIdleHandler : IdleStateHandler(0, 0, HEART_BEAT_TIME) { 
  2.     private val logger = LoggerFactory.getLogger(ClientIdleHandler::class.java) 
  3.     @Throws(Exception::class) 
  4.     override fun channelIdle(ctx: ChannelHandlerContext, evt: IdleStateEvent?) { 
  5.         logInfo(logger,"發(fā)送心跳...."
  6.         ctx.writeAndFlush(HeartBeatPacket())    }    companion object { 
  7.         private const val HEART_BEAT_TIME = 30 
  8.     }} 

ClientLogin:登錄服務(wù)端的 Handler。

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

五. 總結(jié)

這次,我開發(fā)的桌面端程序其實邏輯并不復(fù)雜,只需接收 Web 后臺的指令,然后跟各個設(shè)備進行交互。

接收到 Web 端的指令后,通過 Guava 的 EventBus 將指令通過 TCP 發(fā)送給各個設(shè)備,發(fā)送時需要轉(zhuǎn)化成對應(yīng)的 Packet。因此,核心的模塊就是這個 TCP 自定義的協(xié)議。

責(zé)任編輯:未麗燕 來源: 今日頭條
相關(guān)推薦

2022-06-06 09:28:36

ReactHook

2012-11-19 11:07:42

IBMdw

2015-01-14 15:06:48

定義相機

2016-12-05 17:08:30

tcpsocketandroid

2021-05-29 16:12:00

通信協(xié)議設(shè)備

2009-04-28 13:25:36

Ajax函數(shù)Java

2021-03-09 15:23:45

鴻蒙HarmonyOS應(yīng)用開發(fā)

2021-01-06 05:25:56

項目Springboot應(yīng)用

2023-02-09 08:47:48

TCP協(xié)議FTP

2009-09-13 18:58:07

自定義LINQ提供器

2016-12-05 17:19:10

sockettcpandroid

2022-06-20 08:37:28

接口tokenAO

2017-06-20 12:48:55

React Nativ自定義模塊Note.js

2023-04-04 12:24:10

2023-07-10 07:58:45

2025-09-22 08:03:34

2009-09-03 15:46:57

C#自定義事件

2009-07-07 11:38:54

jsp oracle

2024-08-01 17:20:55

2016-11-08 18:53:08

編譯器
點贊
收藏

51CTO技術(shù)棧公眾號

国产区精品视频| 中文字幕免费精品一区| 成熟丰满熟妇高潮xxxxx视频| 无码国精品一区二区免费蜜桃| 美女精品在线观看| 在线播放国产精品| 91精品人妻一区二区三区四区| 国产精品一二三产区| 久久亚洲春色中文字幕久久久| 国产成人精品亚洲精品| 久久r这里只有精品| 欧洲vs亚洲vs国产| 91精品国产麻豆国产自产在线| 日韩国产一级片| 嫩草在线视频| 91麻豆精品秘密| 亚洲综合日韩在线| 久久久久久久久久一级| 黄色成人av网站| 中文字幕亚洲综合久久| 亚洲永久无码7777kkk| 动漫一区二区三区| 在线看国产一区二区| 欧美精品卡一卡二| free性欧美hd另类精品| 国产欧美一区二区精品忘忧草 | av在线精品| 欧美色播在线播放| bt天堂新版中文在线地址| 成年人在线观看网站| 91视频免费播放| 成人黄视频免费| 国产精品久久久久久免费播放| 久久aⅴ乱码一区二区三区| 欧美高清videos高潮hd| 国产人与禽zoz0性伦| 红桃成人av在线播放| 日韩黄在线观看| 日本一区二区免费视频| japansex久久高清精品| 精品视频全国免费看| aaa毛片在线观看| 黄视频免费在线看| 亚洲18色成人| 人人干视频在线| 欧美黄色视屏| 一区二区三区四区视频精品免费 | 91在线精品观看| 在线观看视频二区| 男女男精品视频| 国产精品劲爆视频| 91麻豆精品在线| 视频一区在线播放| 国产91在线播放九色快色| 国产精品久久久久久久久久久久久久久久久| 欧美日韩国产亚洲一区| 欧美日韩福利电影| 久久久久久久国产精品毛片| 欧美人成在线| 欧美激情亚洲激情| 国产无精乱码一区二区三区| 激情另类综合| 91国产一区在线| 秋霞精品一区二区三区| 久久资源在线| 国产精品综合不卡av| 亚洲天堂视频在线| 国产精品一区二区久久不卡| av在线亚洲男人的天堂| 黑人乱码一区二区三区av| jvid福利写真一区二区三区| 久久99欧美| 国产色在线 com| 国产精品久久久久精k8| 水蜜桃在线免费观看| 欧美黑人猛交的在线视频| 亚洲成人av资源| 青青青在线播放| 国产69精品久久久久9999人| 7777精品伊人久久久大香线蕉超级流畅 | 美女国产在线| 一二三区精品福利视频| 国产网站免费在线观看| av免费在线一区| 欧美一级艳片视频免费观看| 国产一级二级视频| av亚洲在线观看| 欧美日本中文字幕| 国产成人在线免费视频| 久久超级碰视频| 高清视频在线观看一区| 丝袜视频国产在线播放| 欧美激情一区三区| 国产乱子伦精品视频| 678在线观看视频| 欧美羞羞免费网站| 欧美人与性动交α欧美精品| 美女网站色精品尤物极品姐弟| 亚洲色图第三页| 欧美又粗又大又长| 久久一日本道色综合久久| 亚洲mm色国产网站| 青草久久伊人| 亚洲欧洲国产日韩| 欧美成人高潮一二区在线看| 国产成人免费精品| 亚洲国产成人91精品| 色偷偷男人天堂| 精品国产不卡| 欧美xxx视频| 男男激情在线| 久草资源在线视频| 日本成人xxx| 亚洲高清在线播放| 日本人成精品视频在线| 亚洲欧美日韩视频一区| 欧美日韩免费高清一区色橹橹 | 波多野结衣在线观看一区| 久99久精品视频免费观看| 精品日韩电影| 中文字幕在线观看网站| 色欧美日韩亚洲| 91porn在线| 午夜激情久久| 国产精品高潮粉嫩av| 色欲av永久无码精品无码蜜桃| 国产精品久久777777| 欧美日韩亚洲第一| 成人av综合网| 色综合久久88| 国产精品无码一区二区桃花视频| 国产校园另类小说区| 国产在线播放观看| 波多野结衣欧美| 九色91av视频| 精品久久久中文字幕人妻| 国产精品狼人久久影院观看方式| 欧美日韩亚洲第一| 亚洲三级性片| 欧美亚洲视频在线看网址| 好吊色视频一区二区| 亚洲综合图片区| 午夜免费福利网站| 欧美福利电影在线观看| 91情侣偷在线精品国产| 免费黄网站在线播放| 欧美人牲a欧美精品| 一级在线观看视频| 日本亚洲最大的色成网站www| 免费在线成人av| 悠悠资源网亚洲青| 亚洲精品自产拍| 日韩久久中文字幕| 久久亚洲春色中文字幕久久久| 国产特级黄色大片| 亚洲人和日本人hd| 国产精品第三页| 国产视频二区在线观看| 欧美亚洲综合一区| 99热在线观看精品| 国产精品一区二区x88av| 国产精品一二三在线观看| 香蕉免费一区二区三区在线观看| 九色91av视频| 手机看片福利永久| 色系网站成人免费| 精品人体无码一区二区三区| 麻豆91在线观看| 天天爱天天做天天操| 91精品短视频| 91极品视频在线| 男人天堂网在线观看| 欧美少妇性性性| 国产免费美女视频| 成人免费高清视频在线观看| 美女av免费在线观看| 欧美日韩激情在线一区二区三区| 国产精品专区h在线观看| 成人免费高清| 日韩成人在线视频观看| 久久久久久久久久一级| 亚洲精品久久久蜜桃| 日本一卡二卡在线| 日韩电影在线一区| 黄色影视在线观看| 日韩高清成人在线| 国产区亚洲区欧美区| 久草在线视频网站| 亚洲视频电影图片偷拍一区| 国产女人18毛片水18精| 午夜精品福利一区二区三区蜜桃| 亚洲一区二区自偷自拍| 国产精品综合在线视频| 99精品视频播放| 亚洲精品成人| 欧美日韩免费精品| 亚洲国产一区二区三区网站| 日产日韩在线亚洲欧美| 天堂亚洲精品| 国产午夜精品视频免费不卡69堂| 国产精品久久久久久免费 | 日韩在线观看一区二区三区| 欧美一区视频在线| а天堂中文在线官网| 亚洲欧美日韩在线高清直播| 精品国精品国产自在久不卡| 欧亚一区二区三区| 日韩精品一区二区三| 国产精品久久久久一区二区三区 | 在线观看国产一级片| 亚洲美女91| 欧美日韩中文字幕在线播放| 精品日本12videosex| 九九九九九精品| 免费观看性欧美大片无片| 国产成人av网| 免费毛片b在线观看| 九色精品美女在线| 麻豆影院在线| 中文欧美在线视频| 美女做暖暖视频免费在线观看全部网址91 | 久久久久综合网| 天天躁日日躁狠狠躁av麻豆男男 | 亚洲国产美女| 日韩国产精品毛片| 青草国产精品| 欧洲精品在线一区| 午夜精品影视国产一区在线麻豆| 91久久国产自产拍夜夜嗨| 免费在线成人激情电影| 日本在线观看天堂男亚洲 | av资源网在线观看| 亚洲欧美日韩在线高清直播| 四虎电影院在线观看| 日韩电影在线观看中文字幕 | www.欧美免费| av在线三区| 一区二区亚洲精品国产| 玖玖综合伊人| 亚洲区免费影片| 日av在线播放| 亚洲另类欧美自拍| 人成在线免费视频| 亚洲系列中文字幕| 国产精品四虎| 神马久久久久久| 69av亚洲| 久久精品99久久久香蕉| www.欧美日本韩国| 欧美另类在线观看| 久久不射影院| 91爱视频在线| 另类图片综合电影| 国产精品草莓在线免费观看| 日本精品在线中文字幕| 日本午夜精品理论片a级appf发布| 深夜成人影院| 国产精品青青在线观看爽香蕉| 另类中文字幕国产精品| 国产拍精品一二三| 日本少妇精品亚洲第一区| av成人观看| 欧美顶级毛片在线播放| 欧美日韩免费精品| 成人激情在线| 免费成人深夜夜行网站视频| 欧美极品一区二区三区| 日本丰满少妇xxxx| 视频在线在亚洲| 日日干夜夜操s8| 国产精品一区二区无线| 国产大学生视频| 91香蕉视频黄| xxxxx99| 亚洲精品成a人| 国产成人无码精品久在线观看| 色94色欧美sute亚洲线路二 | 日本天堂中文字幕| 亚洲国产cao| 日韩精品一区二区亚洲av观看| 欧美精品一二三| 亚洲精品字幕在线观看| 日韩精品视频在线观看免费| 99青草视频在线播放视| 欧美国产日韩在线| 性国裸体高清亚洲| 国产自产女人91一区在线观看| 亚洲一区二区电影| 久久99蜜桃综合影院免费观看| 欧美电影免费观看高清| 精品无码国模私拍视频| 美国av一区二区| 在线观看国产免费视频| 国产精品久久久一本精品 | 欧美日韩中文一区| 欧美一级视频免费| 最近中文字幕2019免费| 国产中文在线播放| 成人黄色免费看| 希岛爱理av免费一区二区| 在线观看18视频网站| 国产精品久久久亚洲一区| 亚洲日本黄色片| 91美女精品福利| 九九久久免费视频| 欧美制服丝袜第一页| 日韩在线一区二区三区四区| 最新中文字幕亚洲| 亚洲女同志freevdieo| 亚洲最大的成人网| 日本黄色精品| 国产亚洲精品网站| 福利电影一区二区| 美女视频久久久| 日本韩国欧美三级| 亚洲欧美丝袜中文综合| 久久国产精品久久久久久久久久| 最新日韩一区| 欧美性大战久久久久| 亚洲高清二区| 欧美视频亚洲图片| 国产欧美综合色| 国产三级精品三级在线观看| 亚洲黄色在线看| 欧美激情成人动漫| 97se亚洲综合在线| 影音先锋日韩精品| 亚洲免费999| 国产精品午夜电影| 波多野结衣黄色网址| 亚洲精品资源美女情侣酒店| 女厕盗摄一区二区三区| 好看的日韩精品| 影音先锋亚洲电影| 欧美激情 亚洲| 亚洲成精国产精品女| 高清毛片aaaaaaaaa片| 欧美大荫蒂xxx| 亚洲不卡视频| 屁屁影院ccyy国产第一页| 国产一区二区伦理| 成人观看免费视频| 欧美日韩国产123区| 95在线视频| 国产有码在线一区二区视频| 精品国产乱码久久久久久蜜坠欲下 | 内射一区二区三区| 欧美一区二区免费视频| 亚洲大胆人体大胆做受1| 亚洲一区二区自拍| 欧美三级黄美女| 91超薄肉色丝袜交足高跟凉鞋| 一个色妞综合视频在线观看| 日本高清视频免费观看| 51午夜精品视频| 亚洲最大在线| 91日韩视频在线观看| 国产精品午夜在线| 国产免费一区二区三区免费视频| 久久国产精品免费视频| 一区二区三区四区精品视频| 青草青青在线视频| 久久免费电影网| 正在播放木下凛凛xv99| 久久影院中文字幕| 国产精品xxxav免费视频| 国产中文字幕免费观看| 国产日韩欧美a| 国产又色又爽又黄又免费| 欧美日韩成人免费| 丝袜久久网站| 九九热免费在线观看| 亚洲综合色视频| 欧美大片aaa| 成人做爽爽免费视频| 亚洲精品偷拍| 成人黄色a级片| 日韩欧美二区三区| 日韩大片免费观看| 中文字幕久久综合| 成人污视频在线观看| 久久精品99北条麻妃| 美日韩丰满少妇在线观看| 日韩在线麻豆| 一级黄色片在线免费观看| 亚洲成人免费观看| www亚洲人| 国产不卡一区二区三区在线观看| 水蜜桃久久夜色精品一区的特点 | 精品久久久久久亚洲国产300| yw视频在线观看| 国产66精品久久久久999小说| 另类图片国产| 久草视频在线免费看| 国产一区二区三区视频在线观看 | 亚洲精品乱码久久久久久蜜桃91| 国产成人综合网| 五月婷婷激情五月| 国内外成人免费激情在线视频| 色无极亚洲影院|