关于这几天使用IOS的ASYNCSOCKET完成无限后台的过程_ios gcdasyncsocket 进入后台运行-程序员宅基地

技术标签: iOS网络  asyncSocket  心跳包  

这几天用了下ASyncSocket完成前后台即时通讯,当时有想过用消息推送的技术实现的,可是后来想到消息推送的不可靠性还是算了。于是使用了tcp/ip实现后台主动发送数据给前台的功能。

最开始设计后台的时候,我有考虑到数据量比较大的问题,所以数据大的时候我会使用分包和组包的功能去实现。TCP/IP在传输数据的时候,一般不会大于1500字节,所以我每512字节分了

一个包。然后当一次性数据包接收太多的时候,就出现了粘包的问题。因为我在数据传输的时候使用的是json,每一个分包都是由{}括起来的,所以我就想着在包头上加上一段基本不会重复

的分割字符串,然后服务器接收到分包的时候每次都根据这个字符串分割一下,第一次分割的时候第一行绝对是空字符串 例如:@Hinagiku{“Name”=“桂雏菊”}, 我分割出来结果是:

“”,“桂雏菊”,所以说第一行我就可以直接跳过,每次取分包的时候从第二行开始取。然后后台根据包的ID号,序号进行组包。如果当前分包在5分钟内没有接收完毕,就代表当前分包接收失败

了,要求客户端或服务器重新发送。粘包问题解决完毕之后,我开始实现心跳包功能,当时想的是,每隔1分钟发一次心跳包,服务器放一个线程。每隔几秒钟判断一次,当前的所有TCP连接的

最后一次访问时间是多少号,如果超过了这个时间则断开当前连接。

  实现完成之后,我开始着手无限后台功能的实现。在这里我就不说无限后台有哪几种实现方式了,好麻烦,我使用的VOIP模式。我使用的是在code4app上下载的AsyncSocket这个开源类库,

和GCDAsyncSocket的用法是一样的。开始的时候实现的比较轻松,客户端和服务器的连接和数据传输很快就完成了。后来在实现无限后台的时候,出现了一个很奇怪的问题:我在调试状态运行

程序的时候,程序隐藏到后台的时候的确是可以无限运行的,并且TCP连接没有断开。但是我非调试状态运行的时候,TCP连接在3分钟之后就断掉了。 出现这个问题后我找了整整一天的时候,后来

找到了原因,因为我注释掉了一行代码,所以导致TCP连接会被断开。按道理,实现无限后台有以下几个步骤,首先在plist文件中的Required background modes这一项中新增以下两项(默认

项目中是没有这一项的,需要手动添加):App play audio or streams audio/video using AirPlay和App provides Voice over IP services 。IOS7中没有这么麻烦,可以直接点击项目文件,

勾选以下两项:

然后在AsyncSocket.m中,修改以下方法:

 - (BOOL)createStreamsFromNative:(CFSocketNativeHandle)native error:(NSError **)errPtr

- (BOOL)createStreamsToHost:(NSString *)hostname onPort:(UInt16)port error:(NSError **)errPtr

 

拷贝以下代码到这个方法中

CFReadStreamSetProperty(theReadStream, kCFStreamPropertyShouldCloseNativeSocket, kCFBooleanFalse);
CFWriteStreamSetProperty(theWriteStream, kCFStreamPropertyShouldCloseNativeSocket, kCFBooleanFalse);
 CFReadStreamSetProperty(theReadStream, kCFStreamNetworkServiceType, kCFStreamNetworkServiceTypeVoIP);
 CFWriteStreamSetProperty(theWriteStream, kCFStreamNetworkServiceType, kCFStreamNetworkServiceTypeVoIP);

 [(NSInputStream *)theReadStream setProperty:NSStreamNetworkServiceTypeVoIP forKey:NSStreamNetworkServiceType]; //(这里需不需要加上我不清楚,反正加上也不会报错。。。)
 [(NSOutputStream *)theWriteStream setProperty:NSStreamNetworkServiceTypeVoIP forKey:NSStreamNetworkServiceType]; //(这里需不需要加上我不清楚,反正加上也不会报错。。。)

然后在AppDelegate的- (void)applicationDidEnterBackground:(UIApplication *)application这个事件中写入以下代码:

复制代码

- (void)applicationDidEnterBackground:(UIApplication *)application {       BOOL backgroundAccepted = [[UIApplication sharedApplication] setKeepAliveTimeout:600 handler:^{            [self heartbeat];    }];    if (backgroundAccepted)    {        NSLog(@"backgrounding accepted");    } }

复制代码


[self heartbeat] 是我写的一个心跳包的方法,这段代码的意思是:每隔10分钟向服务器发送一次心跳包,保证你的TCP连接是正常的。

由于我以前在applicationDidEnterBackground这个事件中写入了这样的代码,所以才会出现那个奇怪的问题:

 

 

复制代码

- (void)applicationDidEnterBackground:(UIApplication *)application {       BOOL backgroundAccepted = [[UIApplication sharedApplication] setKeepAliveTimeout:600 handler:^{            [self backgroundhandler];    }];    if (backgroundAccepted)    {        NSLog(@"backgrounding accepted");    }      [self backgroundhandler]; }-(void) backgroundinghandler{NSLog(@"### -->backgroundinghandler");        UIApplication*  app = [UIApplication sharedApplication];    bgTask = [app beginBackgroundTaskWithExpirationHandler:^{        //[app endBackgroundTask:bgTask];    }];    dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{        while (1) {            NSLog(@"counter:%d", count++);            NSLog(@"timer:%f", [app backgroundTimeRemaining]);            sleep(1);        }    });}

复制代码

这样的代码其实就是为了向后台借更多的时间,但是我们因为使用VOIP后完全不需要借时间了,在TCP监听到消息的时候,程序会从休眠中唤醒10秒左右,所以说在这10秒内我们把接收到的消息处理完就行了。这是网上教程写出来的一个误区,我们使用

 VOIP的时候完全没有必要再去借时间了。

 

我把下面的这段代码改成了上面的那段代码后,我程序挂起到后台以后,我还是收不到服务器发送到的消息。我感觉我的代码没错,到底是哪里出了错误呢。找了几个小时之后,我无意中看到一段提醒:后台监听消息一定要在真机中运行,在模拟器中是监听

不到的。后来我把测试环境改成了ipad,果然就收到消息了,然后发现我自己的愚蠢。因为ipad拿着太麻烦了,所以我用的是模拟器测试的。 哎,以后大家测试这种代码还是都用真机吧,不要跟我一样被坑了。。。

 

本篇文章差不多也就记录到这里了,我还是一个ios新人,有一些理解可能不正确,希望大家指点出来共同学习。



版权声明:本文为博主原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。
本文链接:https://blog.csdn.net/zhaojian3513012/article/details/46792937

智能推荐

c++ 构造函数详细介绍-程序员宅基地

文章浏览阅读1.1k次,点赞33次,收藏17次。c++构造函数详细介绍

双路比例阀放大器_双比例阀放大板是什么东西-程序员宅基地

文章浏览阅读55次。在液压控制方面,双路比例阀放大器可以将输入的控制信号分成两路,分别控制两个液压阀的开度和流量。这样可以实现更加精确和灵活的液压控制系统,满足不同应用场景的需求。例如,在挖掘机、装载机等重型机械设备中,双路比例阀放大器可以实现多级液压系统的同时控制,提高工作效率和安全性。双路比例阀放大器是一种常见的电子设备,它能够将输入信号放大到所需的水平,并输出两个相等或不同的放大信号。这种放大器通常由一个放大器和一个驱动电路组成,可以用于各种应用中,如液压控制、气动控制等。_双比例阀放大板是什么东西

tensorflow的一些用法(矩阵初始化,变量定义,运算)_python tensorflow 变量 初始化 已知矩阵-程序员宅基地

文章浏览阅读4.8k次。必导入的2个包import tensorflow astfimport numpy asnp #情形1 矩阵运算#构建3*4的0矩阵matrix_1 = np.zeros((3,4))x=tf.constant([[2,2],[2,1]],name='a')y=tf.constant([[0,1],[3,5]],name='b')with tf.Se_python tensorflow 变量 初始化 已知矩阵

pinyin4j 生成 首字母词组组合 多音字词组组合_首字母组合生成器-程序员宅基地

文章浏览阅读1.1k次。目标:词组汉字转拼音首字母组合 (包含多音字)pinyin4j 有些汉字是多音字会生成多个拼音,首先需要去重;取首字母;然后取组合一个词组里面建议不要有超过20个多音字;最好多音字5个左右;如果多音字过多,会产生 翻倍 集合,导致内存泄漏;如果多音字过多,建议把集合里面的 String 换成 StringBuilder;以后有时间在优化/** * 汉字转换位汉语拼音首字母..._首字母组合生成器

mysql 给用户all权限_mysql赋给用户权限grant all privileges on-程序员宅基地

文章浏览阅读1.1k次。查看mysql用户表的结构,Field项都是各类权限限制Host限制登录的IP,User限制登录的用户,Delete_priv限制删除权限,Grant_priv限制权限授予,Super_priv为超级权限,authentication_string为密码的加密字符串grant 权限1,权限2,…权限n on数据库名称.表名称 to 用户名@用户地址 identified by '连接口令';权限..._mysql grant all

使用Druid SQL Parser解析SQL-程序员宅基地

文章浏览阅读1.2w次,点赞9次,收藏50次。在以前的博文《使用Spring Boot JPA Specification实现使用JSON数据来查询实体数据》中讲到了目前业务上的需求就是以前老系统是通过配置SQL去抽取一些业务数据的,但现在新系统想通过页面的一些配置化实现跟配置SQL一样去抽取数据。所以在之前的博文讲到了如何利用JPA Specification和构造的JSON数据去抽取数据。但是老系统很多历史数据都是用SQL去配置的,这些配置当然需要迁移到新系统,但是我们不可能手工一条条去把这些SQL转成当前的JSON结构,这样太浪费时间了。所以我的_druid sql parser

随便推点

微软cl编译器的简单使用_cl编译器tiny模式-程序员宅基地

文章浏览阅读562次。cl的位置:开始 –> 所有程序 -> Microsoft Visual Studio ->Visual Studio Tools -> Developer Command Prompt编译器产生通用对象文件格式 (COFF) 对象 (.obj) 文件。链接器产生可执行文件 (.exe) 或动态链接库文件 (DLL)。注意,所有编译器选项都区分大小写。若要编译但不链接,请使用/_cl编译器tiny模式

设置相机水印字体的大小、位置_平板拍照的水印字体大小-程序员宅基地

文章浏览阅读7.9k次。需求是根据保存的照片的不同分辨率设置不同大小的文字水印,文字需要黑色描边,白底色,位于照片右下角水印描边是非常重要的,两种区别明显的颜色(示例代码中使用的就是黑色和白色)可以有效的提升在不同背景下水印的可识别度。解决方法是获取照片的尺寸,然后根据照片的尺寸进行文字大小的设置;并利用canvas.drawText(test,x,y,paint) 设置文字绘制的位置。根据Bitmap的尺..._平板拍照的水印字体大小

C++ 中的虚:虚基类和虚继承_6-3 沙发床-虚基类-程序员宅基地

文章浏览阅读415次。C++ 中的虚:虚基类和虚继承一、问题由来以沙发床为例:沙发床继承了 沙发 和 床。#include <iostream>#include<string>using namespace std;class Sofa{public: Sofa(float pe = .0, string cr = "black") :price_(pe), color_..._6-3 沙发床-虚基类

UVM——RAL模型基础之一(相关层次、设计流程)_uvm ral-程序员宅基地

文章浏览阅读1.2w次,点赞36次,收藏244次。文章目录一、RAL设计流程二、RAL模型层次相关类三、RAL模型设计实现3.1.利用RAL generator(ralgen)将寄存器描述性文档生成UVM寄存器模型3.1.1.创建寄存器模型类(ralgen自动生成)3.1.2.将寄存器放入register block容器中,并加入到对应的Address Map3.2.创建RAL适配器(adapter)3.3.验证环境中实例化RAL模型并建立连接3..._uvm ral

usb3.1和3.0的区别,usb3.1有什么优点-程序员宅基地

文章浏览阅读2.2w次。从测试结果来看,USB 3.1接口的实际性能要比USB 3.0高出不少,连续读写速度在500MB/s到600MB/s左右,最高成绩甚至可以突破700MB/s。而USB 3.0接口方面,其连续读写速率大概在300MB/s到400MB/s左右。虽然USB 3.1标称的接口理论速率是10Gbps,但是其还保留了部分带宽用以支持其他功能,因此其实际的有效带宽大约为7.2Gbps,理论传输速度应该可..._usb3.1和3.0

【优化路由】基于matlab遗传算法WSN通信路由协议【含Matlab源码 4169期】_无线体域网路由策略matlab-程序员宅基地

文章浏览阅读845次,点赞20次,收藏22次。遗传算法WSN通信路由协议完整的代码,方可运行;可提供运行操作视频!适合小白!_无线体域网路由策略matlab

推荐文章

热门文章

相关标签