最近两个月的时间都忙于赶项目进度,都没有时间好好总结,现在项目基本完成了,就剩下等待审核上线了,不过审核真的是相当坑,等审核结束了,我再写一篇关于审核的文章,各种元数据被拒绝,简直欲哭无泪,不过吃过亏了下次就长记性了,也算是一种学习吧,这边先说一下在项目中遇到的一些问题
自定义TabBarController Push下一级Controller时 会报这样的错误:Unbalanced calls to begin/end appearance transitions for <UIVIewController>
现在的app一般都需要自定义tabbar,这次也不例外,不过在push的过程中居然遇到了这么一个问题,之前都没遇到过,可能是因为自定义的缘故,部分属性需要自己设置,不然会报错,解决方法如下:
自定义了TabBarController 之后必须实现以下
-(void)viewWillAppear:(BOOL)animated
{
[self.selectedViewController beginAppearanceTransition: YES animated: animated];
}
-(void) viewDidAppear:(BOOL)animated
{
[self.selectedViewController endAppearanceTransition];
}
-(void) viewWillDisappear:(BOOL)animated
{
[self.selectedViewController beginAppearanceTransition: NO animated: animated];
}
-(void) viewDidDisappear:(BOOL)animated
{
[self.selectedViewController endAppearanceTransition];
}
判断字符串为空
既然是做项目,那么经常会出现填写注册以及登录信息的,但是很多调皮的用户,经常会用空来输入,我们就需要对空进行判断,空的情况有很多种,之前的话都是通过长度为0进行判断的,不过这样的判断不够全面,经常会有漏网之鱼,所以写了下面的一个判断方法
+ (BOOL) isBlankString:(NSString *)string
{
if (string == nil || string == NULL) {
return YES;
}
if ([string isKindOfClass:[NSNull class]]) {
return YES;
}
if ([[string stringByTrimmingCharactersInSet:[NSCharacterSet whitespaceCharacterSet]] length]==0) {
return YES;
}
return NO;
}
使用信号量来解决代码块执行顺序的问题
代码块是实际开发中经常会使用的,但是代码块的执行顺序问题经常是很让人头疼的,在这次项目中,先要获取到用户的全部信息,然后才能进行相关的订阅主题,但是获取用户的全部信息是一个代码块的方法,每次都没有拿到需要的userid,就开始订阅导致失败,无论是放在代码块中或者代码块外面都不能解决,然后就想了一个信号量+队列的办法来解决,如下所示:
dispatch_semaphore_t signal = dispatch_semaphore_create(0);
dispatch_group_t group=dispatch_group_create();
dispatch_group_async(group, dispatch_get_global_queue(0, 0), ^{
KLRootTabbarController *root=[[KLRootTabbarController alloc]init];
[root acceptfriendList];
});
dispatch_group_notify(group, dispatch_get_global_queue(0, 0), ^{
self.userid = [_topic selectUseridWithUsername:[UserDefaultsUtils valueWithKey:USER_NAME] andFriendname:noti.userInfo[@"FriendUsernamer"]];
dispatch_semaphore_signal(signal);
});
dispatch_semaphore_wait(signal, DISPATCH_TIME_FOREVER);
打开当前app的应用权限管理
app中的设置有一个跳转到本身应用的权限管理界面的功能,实现代码如下:
NSURL * url = [NSURL URLWithString:UIApplicationOpenSettingsURLString];
if([[UIApplication sharedApplication] canOpenURL:url]) {
NSURL*url =[NSURL URLWithString:UIApplicationOpenSettingsURLString];
[[UIApplication sharedApplication] openURL:url];
}
时间戳的转换
出于安全性的考虑,后台说需要使用sessionID,设置的是6分钟超时,每次获取sessionid都会返回给我超时时间以及sessionid,但是后台返回的是时间戳的格式,唉,万恶的后台,安卓常用的时间戳转换好像和ios的相差1000,所以我下面乘上了1000才能够顺利的将时间戳进行转换
-(long long)getDateTimeTOMilliSeconds:(NSDate *)datetime
{
NSTimeInterval interval = [datetime timeIntervalSince1970];
NSLog(@"转换的时间戳=%f",interval);
long long totalMilliseconds = interval*1000 ;
NSLog(@"totalMilliseconds=%llu",totalMilliseconds);
return totalMilliseconds;
}
使用SSL证书连接MQTT服务器
这个问题是最头疼的,困扰了我很久,因为无论是前端还是后台都不了解SSL证书以及MQTT连接,所以只能摸石头过河,连接MQTT还算顺利,通过当前使用的比较多的第三方库进行连接,ssl证书就相对坑很多了,后台给了一个crt证书首先,然后通过钥匙串转换成cer证书,这样我们才能够使用,然后用MQTT第三方库中自带的SSL连接的方法进行连接就可以了(这边和安卓不用,安卓的SSL连接方法和ios不太像,我一开始对着找,就是没找到,然后怎么都连不上,后来翻阅文档才找到了对应的方法)相关代码如下:
MQTT使用ssl的连接方法
self.session = [KLMQTTSingleton shareInstance];
[self.session setSecurityPolicy:[self customSecurityPolicy]];
self.session.delegate = self;
[self.session connectAndWaitToHost:[UserDefaultsUtils valueWithKey:kIP] port:[[UserDefaultsUtils valueWithKey:kPORT] intValue] usingSSL:YES timeout:5];
设置获取证书的方法
- (MQTTSSLSecurityPolicy *)customSecurityPolicy
{
NSString *cerPath = [[NSBundle mainBundle] pathForResource:@"client" ofType:@"cer"];//证书的路径
NSData *certData = [NSData dataWithContentsOfFile:cerPath];
MQTTSSLSecurityPolicy *securityPolicy = [MQTTSSLSecurityPolicy policyWithPinningMode:MQTTSSLPinningModeNone];
securityPolicy.allowInvalidCertificates = YES;
securityPolicy.validatesCertificateChain = YES;
securityPolicy.validatesDomainName = NO;
securityPolicy.pinnedCertificates = @[certData];
return securityPolicy;
}