CocoaAsyncSocket

这个第三方库做了一些底层的封装,让我们在使用的时候可以去关心上层的东西,socket通信就是主机端口,和服务器端口建立连接,进行数据传输,因此整个过程就是服务端持续不断的监听,手机端发送连接请求,服务端接收到连接请求后返回接收到连接请求,再根据业务需求进行进一步通信。

1.一开始我认为socket连接只要做一套tcp的就行,其实是既有基于tcp也有基于udp的,tcp需要经过握手才能连接,udp则可以直接连接。

2.公司的业务需求包含wifi信息的验证,当处于局域网下,路由设备会处于监听状态,当接收到手机端发出的请求连接后会根据设备信息选择回复消息。

3.本义上我认为连接到公司特定的wifi下,那么只需要判断wifi的mac地址即bssid信息就能够判断是否为公司的路由器了,但是存在母路由器连接子路由器这种连接方式,极端情况下母路由和子路由的设备来自不同的厂家,wifi由公司母路由提供,其他品牌的子路由仅仅作为无线ap(hub)存在,这种情况下手机连接到子路由的情况下,依旧是可以控制母路由的,但mac地址自然不是公司提供的,因此不能作为判断是否连接到公司wifi的依据。

4.这种情况的解决只能是与路由器设定协议,路由和手机建立了socket连接,当路由器接收到手机端的请求,就会回复路由器的相关信息,可以返回是否为公司设备最好,如果没有,可以沟通要求返回的信息需要包含路由的mac地址,这样可以再次对比手机端获取的wifi mac地址,如果比对不一致,当然就认为非公司设备。

5.需要特别注意一点,就是路由设备有可能有多台都处于同一局域网下,这时候多台设备都接收到了手机端发起的连接请求,这时候多台设备都会回复信息给手机端,需要对不同的设备进行判断,应该需要添加一些双向验证的机制,来区分设备,从而来控制不同设备的不同需求。

6.重新进行讨论了一下,是这样的流程,手机端向局域网发送连接请求,然后监听组播网段内的回复信息,
公司特定的设备都是会在特定网段的范围内的,
当消息进行回复的时候,
比对bssid和回复的mac地址,一致的选择作为当前连接设备,另外一台设备会添加到设备列表中,当用户切换设备的时候,会进行重新建立UDP-到TCPsocket连接的过程。

7.接下来是对CocoaAsyncsocket的理解。
参考文章:iOS 基于GCDAsyncSocket快速开发Socket通信

根据这个写的UML图
点击查看

注意点:
有人提问说在网络环境以及其他因素下,很有可能会造成客户端或者后端没有接收到回调或者请求,那该怎么办呢?

我们需要加上消息回执的处理

客户端发出请求的时候,可以将该请求放到存到数组里面,等到后端的相应回调,如果该请求超时,说明后端没有接收到我们的请求,我们可以将该请求重新发送

客户端接收请求的时候,后端将数据发给客户端,客户端需要增加回执处理,告诉后端,客户端接收到数据了,如果后端没接收到,也重新推一遍数据,客户端和后端双向保护解决丢失问题

然后需要注意处理socket会出现不断断开的情况。

接下来要处理socket的进入后台就断开连接,进入前台重新开始连接的情况。因为socket保持长期后台是不能通过苹果审核的,如果真有需要,以推送的方式实现。

socket的粘包。半包处理

消息=消息头+消息体。消息头用于描述消息本身的基本信息,消息体则为消息的具体内容
这里写图片描述

如上图所示,假如我们的一个消息是这么定义的

消息头 = msgId(4B)+version(2B)+len(4B),共占用10字节

消息体 = len中描述的16字节长

所以这条消息的长度就是 26字节

可以看到,要想知道一条完整数据的边界,关键就是消息头中的len字段

假如我们现在接收到的数据是这样的:
这里写图片描述

这个情况下即包含了粘包,也出现了半包的情况,三个数据包粘在了一起,最后一个数据包没有接收完全,出现了半包的情况,看看代码如何处理

- (void)onSocket:(AsyncSocket *)sock didReadData:(NSData *)data withTag:(long)tag 
{ 
  while (_readBuf.length >= 10)//因为头部固定10个字节,数据长度至少要大于10个字节,我们才能得到完整的消息描述信息 
  { 
    NSData *head = [_readBuf subdataWithRange:NSMakeRange(0, 10)];//取得头部数据 
    NSData *lengthData = [head subdataWithRange:NSMakeRange(6, 4)];//取得长度数据 
    NSInteger length = [[[NSString alloc] initWithData:lengthData encoding:NSUTF8StringEncoding] integerValue];//得出内容长度 
    NSInteger complateDataLength = length + 10;//算出一个包完整的长度(内容长度+头长度) 
    if (_readBuf.length >= complateDataLength)//如果缓存中数据够一个整包的长度 
    { 
      NSData *data = [_readBuf subdataWithRange:NSMakeRange(0, complateDataLength)];//截取一个包的长度(处理粘包) 
      [self handleTcpResponseData:data];//处理包数据 
      //从缓存中截掉处理完的数据,继续循环 
      _readBuf = [NSMutableData dataWithData:[_readBuf subdataWithRange:NSMakeRange(complateDataLength, _readBuf.length - complateDataLength)]]; 
    } 
    else//如果缓存中的数据长度不够一个包的长度,则包不完整(处理半包,继续读取) 
    { 
      [_socket readDataWithTimeout:-1 buffer:_readBuf bufferOffset:_readBuf.length tag:0];//继续读取数据 
      return; 
    } 
  } 
  //缓存中数据都处理完了,继续读取新数据 
  [_socket readDataWithTimeout:-1 buffer:_readBuf bufferOffset:_readBuf.length tag:0];//继续读取数据 
}
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值