想要实现在内网的环境下(完全切断外网),实现后台给前台推送通知,前台能够收到通知的需求,就不能够使用苹果的APNS了,要在前台和后台之间建立一个长连接,并且保证APP在手机前台和后台的时候都能保持这个长连接不断掉。
选择用WebSocket来做这个长连接,WebSocket是对Socket做了一次封装,在代码中选择使用facebook出的SocketRocket第三方库来完成WebSocket长连接的搭建。SocketRocket的下载地址为:https://github.com/facebook/SocketRocket。
1、在项目中引入SocketRocket第三方;
2、在AppDelegate.m文件中引入头文件"SRWebSocket.h";
3、在AppDelegate.m文件中写上协议<SRWebSocketDelegate>;
4、在AppDelegate.m文件中属性化SRWebSocket:
@property(nonatomic,strong)SRWebSocket*webSocket;
5、在AppDelegate.m文件中写上连接WebSocket的方法,并且在合适的地方调用这个方法:
#pragma mark - 连接WebSocket
-(void)connectWebSocket
{
self.webSocket.delegate=nil;
[self.webSocketclose];
NSString*token = [UserDatauserInfo].token;
NSString*ucOrganizationId = [NSStringstringWithFormat:@"%@", [[NSUserDefaultsstandardUserDefaults]objectForKey:@"schoolCode"]];
NSString*requestStr = [NSStringstringWithFormat:@"ws://%@/connect/ios/%@/%@", [[NSUserDefaultsstandardUserDefaults]objectForKey:@"networkAddress"], ucOrganizationId, token];
self.webSocket= [[SRWebSocketalloc]initWithURLRequest:[NSURLRequestrequestWithURL:[NSURLURLWithString:[requestStrURLDecodedString]]]];
self.webSocket.delegate=self;
[self.webSocketopen];
}
上面的requestStr是和后台约定好的地址。
6、在AppDelegate.m文件中撰写SRWebSocket的代理方法:
#pragma mark - SRWebSocketDelegate
/**
* 连接成功会调用此方法;
* 在此方法中开启一个子线程,在子线程中向后台发送心跳包。
*/
- (void)webSocketDidOpen:(SRWebSocket*)webSocket
{
NSLog(@"webSocket连接成功");
[UIViewtoast:@"内网连接成功!"];
//隐式创建子线程(这种方式创建子线程的开销小)
[selfperformSelectorInBackground:@selector(creatTimer)withObject:nil];
}
/**
* 连接失败会调用此方法;
* 在此方法种中重连webSocket。
*/
- (void)webSocket:(SRWebSocket*)webSocket didFailWithError:(NSError*)error
{
NSLog(@"webSocket连接失败,失败原因:%@", error);
NSString*message = [NSStringstringWithFormat:@"内网连接失败,失败原因为:%@", error];
[UIViewtoast:message];
//重连webSocket
[selfreconnectionWebSocket];
}
/**
* webSocket连接被关闭就会调用此方法;
* 在此方法中重新连接webSocket。
*/
- (void)webSocket:(SRWebSocket*)webSocket didCloseWithCode:(NSInteger)code reason:(NSString*)reason wasClean:(BOOL)wasClean
{
NSLog(@"webSocket连接被断开,断开的编码为:%zd,断开的原因为:%@", code, reason);
NSString*message = [NSStringstringWithFormat:@"内网被断开,断开的编码为:%zd,断开的原因为:%@", code, reason];
[UIViewtoast:message];
//关闭定时器
[selfstopTimer];
[selfreconnectionWebSocket];
}
/**
* 在此方法中接收后台传过来的消息;
* 只要接收到后台传过来的消息就让它以alert的形式展示出来。
*/
- (void)webSocket:(SRWebSocket*)webSocket didReceiveMessage:(id)message
{
NSLog(@"接收到后台推送过来的消息:%@", message);
//收到消息后向后台发送一个反馈
[self.webSocketsend:@"received message"];
}
/**
* 心跳包:一般情况下建立长连接的时候都会建立一个心跳包,用于每隔一段时间通知一次服务端,客户端还是在线,用以保活;
* 每隔一段时间可以利用webSocket的sendPing:方法向后台发送ping,作为心跳包来告诉后台前台还存在,后台收到ping之后就会返回pong,给前台一个回应,告诉前台后台还存在;
* 后台收到ping之后返回pong的时候,前台就会调用下面的代理方法;
* 前台利用sendPing:方法发送ping的时候,后台只能在网络架构的底层接收到ping,后台人员不能捕捉到ping,返回pong也是网络架构自动做出的反应,不是后台人员能控制的,所以一般不用ping作为心跳包。
*/
- (void)webSocket:(SRWebSocket*)webSocket didReceivePong:(NSData*)pongPayload
{
NSString*reply = [[NSStringalloc]initWithData:pongPayloadencoding:NSUTF8StringEncoding];
NSLog(@"%@",reply);
}
7、在AppDelegate.m文件中撰写其他的需要调用的方法:
#pragma mark - 创建定时器
/**
* 在子线程中创建定时器;
* 每隔5秒向后台发送一次心跳包。
*/
-(void)creatTimer
{
NSLog(@"创建定时器,%@", [NSThreadcurrentThread]);
self.childThreadTimer= [NSTimerscheduledTimerWithTimeInterval:10.0ftarget:selfselector:@selector(sendHeartBeat)userInfo:nilrepeats:YES];
[[NSRunLoopcurrentRunLoop]run];
}
#pragma mark - 向后台发送心跳包
/**
* 利用webSocket的send:方法向后台发送心跳包,KA是与后台约定好的字符串。
*/
-(void)sendHeartBeat
{
NSLog(@"向后台发送心跳包,%@", [NSThreadcurrentThread]);
// NSData *data = [@"test" dataUsingEncoding:NSUTF8StringEncoding];
// [self.webSocket sendPing:data];
if(self.webSocket)
{
[self.webSocketsend:@"KA"];
}else
{
return;
}
}
#pragma mark - 关闭定时器
/**
* 当退出内网的时候需要关闭定时器并且断开webSocket。
*/
-(void)stopTimer
{
[self.childThreadTimerinvalidate];
//关闭webSocket
self.webSocket.delegate=nil;
[self.webSocketclose];
self.webSocket=nil;
}
#pragma mark - 重连WebSocket
/**
* WebSocket如果无线次数重连的话会导致UI不断刷新,所以WebSocket最多重连五次。
*/
-(void)reconnectionWebSocket
{
staticintcount =0;
count++;
if(count <5)
{
self.webSocket=nil;
[selfconnectWebSocket];
}
}
由于本人是第一次在简书上面发东西,可能在撰写此篇文章的时候格式上面有些不工整,容我慢慢研究一下简书的撰写方法,以后加强一下文章的易读性。各位读者如果有印象笔记账号的话可以参考我印象笔记上面的此篇文章,那上面的易读性会好一些,连接地址:https://app.yinxiang.com/shard/s10/nl/18104452/1aa5be40-28ed-46b8-83e1-45033044a881/。
本人是开发小白,技术难免有不周全之处,还请各位读者批评指正,相互讨论。