[iOS] 接入WebSocket记录 + 一些个人经验

作者: ck2016 | 来源:发表于2016-10-25 18:58 被阅读4020次

闲扯

WebSocket 以前没用过,之前写过一篇博客是基于原生socket的(查看)比较复杂,慎入。今天另外一个APP需要接websocket了,然后便找到了facebook的 SocketRocket 框架,然后用了一天时间接上了,完成了掉线自动重连,自动重登录,心跳等等功能,用法比原生socket简单(原生socket基于TCP/UDP协议)。

为什么用 WebSocket

因为APP里面有个聊天功能,需要服务器主动推数据到APP。HTTP 通信方式只能由客户端主动拉取,服务器不能主动推给客户端,如果有实时的消息,要立刻通知客户端就麻烦了,要么客户端每隔几秒钟发一次请求,看看有没有新数据,这种方式想想都知道耗流量电量。还一种方式就是走TCP/UDP协议服务器主动推给你,这种方式省流量。还有就是用websocket,websocket是h5里面的东西,h5我不太会,反正它比原生socket用法简单。

用法

用 SocketRocket 框架,记住几个代理方法就好了,很简单。
1.创建和设置代理对象

SRWebSocket *socket = [[SRWebSocket alloc] initWithURLRequest:
[NSURLRequest requestWithURL:[NSURL URLWithString:@"http://ip地址:端口"]];

socket.delegate = self;    // 实现这个 SRWebSocketDelegate 协议啊

[socket open];    // open 就是直接连接了

2.连接成功会调用这个代理方法

- (void)webSocketDidOpen:(SRWebSocket *)webSocket {
    NSLog(@"连接成功,可以立刻登录你公司后台的服务器了,还有开启心跳");
}

3.连接失败会调用这个方法,看 NSLog 里面的东西

- (void)webSocket:(SRWebSocket *)webSocket didFailWithError:(NSError *)error {
    NSLog(@"连接失败,这里可以实现掉线自动重连,要注意以下几点");
    NSLog(@"1.判断当前网络环境,如果断网了就不要连了,等待网络到来,在发起重连");
    NSLog(@"2.判断调用层是否需要连接,例如用户都没在聊天界面,连接上去浪费流量");
    NSLog(@"3.连接次数限制,如果连接失败了,重试10次左右就可以了,不然就死循环了。
    或者每隔1,2,4,8,10,10秒重连...f(x) = f(x-1) * 2, (x<5)  f(x)=10, (x>=5)");
}

4.连接关闭调用这个方法,注意连接关闭不是连接断开,关闭是 [socket close] 客户端主动关闭,断开可能是断网了,被动断开的。

- (void)webSocket:(SRWebSocket *)webSocket didCloseWithCode:(NSInteger)code reason:(NSString *)reason wasClean:(BOOL)wasClean {
    NSLog(@"连接断开,清空socket对象,清空该清空的东西,还有关闭心跳!");
}

5.收到服务器发来的数据会调用这个方法

- (void)webSocket:(SRWebSocket *)webSocket didReceiveMessage:(id)message  {
    NSLog(@"收到数据了,注意 message 是 id 类型的,学过C语言的都知道,id 是 (void *)  
        void* 就厉害了,二进制数据都可以指着,不详细解释 void* 了");
    NSLog(@"我这后台约定的 message 是 json 格式数据
        收到数据,就按格式解析吧,然后把数据发给调用层");
}

6.向服务器发送数据
发送的时候可能断网,可能socket还在连接,要判断一些情况,写在下面了
发送逻辑是,我有一个 socketQueue 的串行队列,发送请求会加到这个队列里,然后一个一个发出去,如果掉线了,重连连上后继续发送,对调用层透明,调用层不需要知道网络断开了

- (void)sendData:(id)data {
    WEAKSELF(ws);
    dispatch_async(self.socketQueue, ^{
        if (ws.socket != nil) {
            // 只有 SR_OPEN 开启状态才能调 send 方法啊,不然要崩
            if (ws.socket.readyState == SR_OPEN) {
                [ws.socket send:data];    // 发送数据

            } else if (ws.socket.readyState == SR_CONNECTING) {
                NSLog(@"正在连接中,重连后其他方法会去自动同步数据");
                // 每隔2秒检测一次 socket.readyState 状态,检测 10 次左右
                // 只要有一次状态是 SR_OPEN 的就调用 [ws.socket send:data] 发送数据
                // 如果 10 次都还是没连上的,那这个发送请求就丢失了,这种情况是服务器的问题了,小概率的
                // 代码有点长,我就写个逻辑在这里好了

            } else if (ws.socket.readyState == SR_CLOSING || ws.socket.readyState == SR_CLOSED) {
                // websocket 断开了,调用 reConnect 方法重连
                [ws reConnect:^{
                    NSLog(@"重连成功,继续发送刚刚的数据");
                    [ws.socket send:data];
                }];
            }
        } else {
            NSLog(@"没网络,发送失败,一旦断网 socket 会被我设置 nil 的");
            NSLog(@"其实最好是发送前判断一下网络状态比较好,我写的有点晦涩,socket==nil来表示断网");
        }
    });
}

7.心跳机制
心跳机制就不难了,开个定时器,问下后台要每隔多少秒发送一次心跳请求就好了。然后注意,断网了或者socket断开的时候把心跳关一下,省资源,不然都断网了,还在循环发心跳,浪费CPU和电量。

8.终于接完websocket了,下班回家压压惊。我第一次用,其实不难,就是考虑的情况比较多,整个逻辑有点多,主要代码就是上面那些了,其他不重要的代码我就不复制粘贴上来了。

相关文章

网友评论

  • 3d31ae4542df:你好 调用close 主动关闭连接 为什么还能收到数据 怎么才能真正主动关闭 不收到数据的那种
    ck2016:@w天涯过客 客户端关了也收不到吧
    3d31ae4542df:已经搞定了 是后台没有去 关闭
    ck2016:@w天涯过客 有点奇怪,得调试一下
  • 丿破灬孩:退到后台之后怎么才能一直受到消息 我这边一会能收到 一会收不到
    ck2016:@丿破灬孩 看下iOS后台保活
  • 红街咖啡:请问一下怎么手动导入啊,现在这个框架手动好像无法导入
    ck2016:@红街咖啡 github上下载,然后拖进去啊
  • Louis_hey:app推到后台和杀死进程怎么保持长连接,收到消息
    Louis_hey:@ck2016 嗯,就是走asp推送通知栏消息,点击去在重新连接
    ck2016:@Louis_hey 杀进程肯定走 apns推送了
  • Louis_hey:有没有Demo
    ck2016:@Louis_hey 对啊,开个timer,定时发就行
    Louis_hey:@ck2016 心跳机制是怎么弄的,能否较详细的说明下,是要每隔固定时间一直向服务器发送么
    ck2016:@Louis_hey 没哦
  • f105fd19438c::smile: 赞,成功避免了几个坑,十分感谢
    ck2016:@Bakakun :+1: :+1: :+1: :+1:
  • 69441e588ed0:您好,请问一下,SRWebSocketyou注册机制吗? 类似那种订阅功能,接收特定的消息机制?
    ck2016:@LG天空之城 没有哦,zoreMQ 有
  • Louis_hey:发送消息成功有回调么?
    ck2016:@Louis_hey 对
    Louis_hey:@ck2016 这个是后台要返回吧
    ck2016:@Louis_hey 我忘了,要回调可以自己加一个
  • 风之谷等风来:连接服务器的时候总是失败,2133错误码
  • 正义必胜biu:第6步重连 那些参数不太明白啊。可以具体告诉我一下吗
    ck2016:@正义ing 哪些
  • paradisery:写的不错,跟我踩的坑一致,早看到你写的,就不用踩坑了,😌,连接是失败里的逻辑写的不错。断线重发机制也不错:grin:
    ck2016:@paradisery :grin:
  • fc18f69e6ff0:你好,心跳连接怎么实现我找了一下没有找到openTimer的方法
    ck2016:@ShenYj 是的
    ShenYj:@ck2016 不仅仅是App开个定时器发送心跳包就可以吧,是不是后台也要做相应对接,我看说是后台发送心跳包,移动端接收到后返回用以判断应用是否连接,另外增加心跳后,能否达到应用后台保活的效果?
    ck2016:@闭上眼睛 开个定时器,循环发心跳包就行了
  • 民谣里不是故事就是诗丶:我这边open成功了 send一条数据 后台也收到了 但是会自动自动close
    触发- (void)webSocket:(SRWebSocket *)webSocket didCloseWithCode:(NSInteger)code reason:(NSString *)reason wasClean:(BOOL)wasClean;方法是什么情况?
    求教学~我并没有主动调用close
    民谣里不是故事就是诗丶:@ck2016 查到原因了 是后台在不该return的时候return了 导致销毁了对象( ¯ᒡ̱¯ )و
    ck2016:@民谣里不是故事就是诗丶 没在后台登录,服务器应该会把你踢掉的
    ck2016:@民谣里不是故事就是诗丶 本地端程序被关了,断网了会主动close,后端的机制它把socket断开了,你也会被 close
  • 仁一:[webSocket open]方法好像只能调用一次,断网了重新连接要怎么做?
    ck2016:@仁一 整个socket对象不要了,重新 alloc 一个再连接
  • 电子邮箱没有箱:请问一下,开启心跳就是重新执行[webSocket open]方法吗?
    ck2016:@电子邮箱没有箱 扣扣657668857
    电子邮箱没有箱: @ck2016 方便留个联系方式不
    ck2016:@电子邮箱没有箱 这个是打开socket的哦,开启心跳有个 openTimer 的方法好像
  • Ryan文濤:赞赞赞!!!
    ck2016:@Ryan文濤 :grin:
  • 圣罗迦奈:好东西 笑纳啦 :blush:
    ck2016:@圣罗迦奈 :grin:
  • Rchongg::+1:
    ck2016:@Rchongg :relaxed:
  • 韦韦韦:赞一个
    ck2016:@韦韦韦 谢谢啊 :grin:

本文标题:[iOS] 接入WebSocket记录 + 一些个人经验

本文链接:https://www.haomeiwen.com/subject/bhlhuttx.html