iOS日常捡漏

1.NSRunLoop

今天做项目的时候,要想开启一个计时器监控系统的时间,用到了NSRunLoop,一般来说,是用CFRunLoopGetCurrent这个函数来获取当前线程的runLoop的。但是如果应用在没有开启子线程的情况下,那么当前线程也就是主线程。也就是说CFRunLoopGetCurrent函数获取的runLoop()是当前线程的,也就是主线程的。在这种情况下,一旦run后,根本关闭不了,因为获取到的是的runLoop(假设是A),是主线程的runLoop。而你想要关闭的是子线程的runLoop(假设是B),根本都没有获取到子线程的runLoop。所以关闭的代码并没有起到任何作用。再说,也没有权限停止主线程的runLoop,在程序没有退出之前,是没法关闭停止主线程的。

参考文献:NSThread 、NSRunLoop 和 Dispatch Queue - kmyhy的专栏 - 博客频道 - CSDN.NET

后来的做法是:新开了一个线程(子线程),每个线程中都自带RunLoop,用CFRunLoopGetCurrent()函数可以获得当前线程的RunLoop。(这样就可以保证获取的是不会是主线程的runLoop)

ps:如果RunLoop中没有任何源,RunLoop不会运行。我们可以为这个RunLoop添加新的源。

iOS日常捡漏_第1张图片
开启子线程
在子线程开启计时


iOS日常捡漏_第2张图片
停止子线程的runLoop

2.

会发现页面跳转不了,并出现这些警告:Only run on the main thread

原因:真机环境下,UI的处理必须是单线程的,所以当 在一个已存在的UI上切换新的UI时,会出现Only run on the main thread 错误。告诉你,只能在主线程上运行。

因为我开启的是一个子线程,利用子线程监听到一定的时间后,就进行账号主线,从主菜单页面强行跳转到登录页面。所以产生这样的警告。

解决办法:在单线程里面操作,进行跳转:dispatch_async(dispatch_get_main_queue(), ^{

LoginViewController *loginController = [[LoginViewController alloc]init];

[self presentViewController:loginController animated:NO completion:nil];

});

iOS日常捡漏_第3张图片

3.

尽管如此,你会发现还是无法跳转,会出现这样的警告:

Warning: Attempt to present on   which is already presenting (null)

出现这样的警告,首先是因为我从主菜单页面(mainMenuController)到登陆页面(loginViewController)不是push过去的,而是presenting过去的。而此时在主菜单页面上刚好有一个选择的弹框。如下图:


iOS日常捡漏_第4张图片
弹框的图片

而这个弹框也是presenting到主菜单页面的。所以必须先要关掉(diss)这个弹框,才能跳转到登录页面。如下图:

iOS日常捡漏_第5张图片
dismiss的图片

先关掉这个,就可以了。 

你可能感兴趣的:(iOS日常捡漏)