ios:NSRunLoop

ios:NSRunLoop

1.NSRunLoop是消息机制的处理模式

NSRunLoop的作用在于有事情做的时候使的当前NSRunLoop的线程工作,没有事情做让当前NSRunLoop的线程休眠

2.nstimer默认添加到当前NSRunLoop中,也可以手动制定添加到自己新建的NSRunLoop的中

[NSTimer schduledTimerWithTimeInterval: target:selector:userInfo:repeats];

此方法默认添加到当前NSRunLoop中

NSTimer *timer = [NSTimer timerWithTimeInterval: invocation:repeates:];

NSTimer *timer = [[NSTimer alloc] initWithFireDate:...];

创建timer  [[NSRunLoop currentRunLoop] addTimer:timer forMode:NSRunLoopCommonModes];

注意 timer的释放

3.NSRunLoop就是一直在循环检测,从线程start到线程end,检测inputsource(如点击,双击等操作)同步事件,检测timesource同步事件,检测到输入源会执行处理函数,首先会产生通知,corefunction向线程添加runloop observers来监听事件,意在监听事件发生时来做处理。

4.runloopmode是一个集合,包括监听:事件源,定时器,以及需通知的runloop observers

模式包括:

default模式:几乎包括所有输入源(除NSConnection) NSDefaultRunLoopMode模式

mode模式:处理modal panels

connection模式:处理NSConnection事件,属于系统内部,用户基本不用

event tracking模式:如组件拖动输入源 UITrackingRunLoopModes 不处理定时事件

common modes模式:NSRunLoopCommonModes 这是一组可配置的通用模式。将input sources与该模式关联则同时也将input sources与该组中的其它模式进行了关联。

每次运行一个run loop,你指定(显式或隐式)run loop的运行模式。当相应的模式传递给run loop时,只有与该模式对应的input sources才被监控并允许run loop对事件进行处理(与此类似,也只有与该模式对应的observers才会被通知)

例:

1).在timer与table同时执行情况,当拖动table时,runloop进入UITrackingRunLoopModes模式下,不会处理定时事件,此时timer不能处理,所以此时将timer加入到NSRunLoopCommonModes模式(addTimer forMode)

2).在scroll一个页面时来松开,此时connection不会收到消息,由于scroll时runloop为UITrackingRunLoopModes模式,不接收输入源,此时要修改connection的mode

[scheduleInRunLoop:[NSRunLoop currentRunLoop]forMode:NSRunLoopCommonModes];

5.关于-(BOOL)runMode:(NSString *)mode beforeDate:(NSDate *)date;方法

指定runloop模式来处理输入源,首个输入源或date结束退出。

暂停当前处理的流程,转而处理其他输入源,当date设置为[NSDate distantFuture](将来,基本不会到达的时间),所以除非处理其他输入源结束,否则永不退出处理暂停的当前处理的流程。

6.while(A){

[[NSRunLoop currentRunLoop] runMode:NSDefaultRunLoopMode beforeDate:[NSDate distantFuture]];

}

当前A为YES时,当前runloop会一直接收处理其他输入源,当前流程不继续处理,出为A为NO,当前流程继续

7.perform selector在thread中被序列化执行,这样就缓和了许多在同一个thread中运行多个方法所产生的同步问题。perform selector source在运行完selector后自动从run loop中移除。

当在非main thread中perform selector时,其thread中必须有一个激活的run loop。对于你自己创建的thread而言,只有你的代码显式的运行一个run loop后该perform selector才能得到执行。Run loop在当loop运行时处理所有已排队的perform selector,而不是在一个loop循环时只处理某一个perform selector。

8.performSelector关于内存管理的执行原理是这样的执行 [self performSelector:@selector(method1:) withObject:self.tableLayer afterDelay:3]; 的时候,系统会将tableLayer的引用计数加1,执行完这个方法时,还会将tableLayer的引用计数减1,由于延迟这时tableLayer的引用计数没有减少到0,也就导致了切换场景dealloc方法没有被调用,出现了内存泄露。

利用如下函数:

[NSObject cancelPreviousPerformRequestsWithTarget:self]

当然你也可以一个一个得这样用:

[NSObject cancelPreviousPerformRequestsWithTarget:self selector:@selector(method1:) object:nil]

加上了这个以后,顺利地执行了dealloc方法

在touchBegan里面

[self performSelector:@selector(longPressMethod:) withObject:nil afterDelay:longPressTime]

然后在end 或cancel里做判断,如果时间不够长按的时间调用:

[NSObject cancelPreviousPerformRequestsWithTarget:self selector:@selector(longPressMethod:) object:nil]

取消began里的方法

**********************************以下是我在cocoachina中看到的一份总结 转载过来

线程实现的几种方式:
1. Operation Objects   // NSOperation及相关子类
2. *****                           // dispatch_async等相关函数
3. Idle-time notifications  //  NSNotificationQueue,低优先级
3. Asynchronous functions  // 异步函数
4. Timers                      // NSTimer
5. Separate processes  // 没用过

线程创建的成本:
kernel data structures  约1KB
Stack space             512KB(secondary threads) 
                                   1MB(iOS main thread)
Creation time           约90 microseconds

Run Loop和线程的关系:
1. 主线程的run loop默认是启动的,用于接收各种输入sources
2. 对第二线程来说,run loop默认是没有启动的,如果你需要更多的线程交互则可以手动配置和启动,如果线程执行一个长时间已确定的任务则不需要。

Run Loop什么情况下使用:
a. 使用ports 或 input sources 和其他线程通信   // 不了解
b. 在线程中使用timers                                             // 如果不启动run loop,timer的事件是不会响应的 
c. 在Cocoa 应用中使用performSelector...方法   // 应该是performSelector...这种方法会启动一个线程并启动run loop吧
d. 让线程执行一个周期性的任务                            // 如果不启动run loop, 线程跑完就可能被系统释放了

注:timer的创建和释放必须在同一线程中。
[[NSRunLoop currentRunLoop] addTimer:timer forMode:NSRunLoopCommonModes];  此方法会retain timer对象的引用计数。

////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////

如下是一个CFRunLoop的demo 帮助理解其使用:

/!!!如果没有如下的currentLoop 那么不会执行initPlayer方法中的timer 即执行完playerThread便结束
-
(void) playerThread:(id)unused
{
    currentLoop =
CFRunLoopGetCurrent();//子线程的runloop引用
  
 
    NSAutoreleasePool *pool = [[NSAutoreleasePool
alloc] init];//为子线程创建自动释放池
    
    [self
initPlayer];
    
   
CFRunLoopRun();
    
    [pool
release];
}

-(void) initPlayer
{
    //
在这里你可以初始化一个工作类,比如声音或者视频播放
    [NSTimer
scheduledTimerWithTimeInterval:3.0
target:self
                                                         
selector:@selector(checkState:) userInfo:nil repeats:YES];
}

-(void)
checkState:(NSTimer*) timer
{
   
//需要退出自定义的线程了
    //if()
   
//{
       
//释放子线程里面的资源
       
//释放资源的代码....
        /*结束子线程任务
CFRunLoopStop,This function forces rl to stop running
and
         return control to the
function that called CFRunLoopRun or
CFRunLoopRunInMode
         for the
current run loop activation. If the run loop is nested with a
callout
         from one activation
starting another activation running, only the
innermost
         activation is
exited.*/
       
//CFRunLoopStop(currentLoop);
    //}
  
 
    for(int i = 0 ; i < 100;
i++)
    {
       
NSLog(@"%d", i);
    }
  
 
    //CFRunLoopStop 会将当前timer结束掉 如果runloop不结束
timer会反复执行 
   
CFRunLoopStop(currentLoop);
  
 
}

-(void)click:(id)sender{
    NSThread
*thread = [[NSThread alloc] initWithTarget:self selector:@selector(playerThread:
) object:nil];
    
   
//开启线程,如果是利用NSOperation,只需要加入到NSOperationQueue里面去就好了,queue自己会在合适的时机执行线程,而不需要程序员自己去控制。
   
[thread start];
    
}

-
(void)viewDidLoad
{
    [super
viewDidLoad];
    
    [self.view
setBackgroundColor:[UIColor whiteColor]];
  
 
    UIButton *button = [[UIButton alloc]
initWithFrame:CGRectMake(10, 10, 100, 50)];
    [button
setBackgroundColor:[UIColor blueColor]];
    [button
setTitle:@"abc" forState:UIControlStateNormal];
    [button
setTitleColor:[UIColor whiteColor]
forState:UIControlStateNormal];
    [button
setTitleColor:[UIColor redColor]
forState:UIControlStateHighlighted];
    [button
setBackgroundColor:[UIColor blueColor]];
    [button
addTarget:self action:@selector(click:)
forControlEvents:UIControlEventTouchDown];
    [self.view
addSubview:button];
    [button release];
  
 
}

ios:NSRunLoop,布布扣,bubuko.com

时间: 2024-11-10 01:33:25

ios:NSRunLoop的相关文章

iOS NSRunLoop那些事

iOS中timer相关的延时调用,常见的有NSObject中的performSelector:withObject:afterDelay:这个方法在调用的时候会设置当前runloop中timer,还有一种延时,直接使用NSTimer来配置任务. 这两种方式都一个共同的前提,就是当前线程里面需要有一个运行的runloop并且这个runloop里面有一个timer. 我们知道:只有主线程会在创建的时候默认自动运行一个runloop,并且有timer,普通的子线程是没有这些的.这样就带来一个问题了,有

iOS开发中NSRunloop跟NSTimer的问题

在Windows时代,大家肯定对SendMessage,PostMessage,GetMessage有所了解,这些都是windows中的消 息处理函数,那对应在ios中是什么呢,其实就是NSRunloop这个东西.在ios中,所有消息都会被添加到NSRunloop中,分为'input source'跟'timer source'种,并在循环中检查是不是有事件需要发生,如果需要那么就调用相应的函数处理. 我们在使用NSTimer的时候,可能会接触到runloop的概念, 下面是一个简单的例子 -

[iOS]浅谈NSRunloop工作原理和相关应用

一. 认识NSRunloop  1.1 NSRunloop与程序运行 那么具体什么是NSRunLoop呢?其实NSRunLoop的本质是一个消息机制的处理模式.让我们首先来看一下程序的入口——main.m文件,一个ios程序启动后,只有短短的十行代码居然能保持整个应用程序一直运行而没有退出,是不是有点意思?程序之所以没有直接退出是因为UIApplicationMain这个函数内部默认启动了一个跟主线程相关的NSRunloop对象,而UIApplicationMain这个函数一直执行没有返回就保存

IOS 等待条件满足再向下执行但不主卡线程NSRunLoop

当我们需要等一个异步的结果才能向下执行代码,写回调又很麻烦,功能相对简单的时候,就可以插入以下红色部分的代码. - (IBAction)start:(id)sender{pageStillLoading = YES;[NSThread detachNewThreadSelector:@selector(loadPageInBackground:)toTarget:self withObject:nil];[progress setHidden:NO];while (pageStillLoadin

IOS OC 多任务定时器 NSRunLoop 管理 NSTimer

下面有两种做法 1.使用日期组件 NSDateComponents 2.使用NSString 生成一个日期 //  创建一个日历对象 NSCalendar *calendar = [NSCalendar currentCalendar]; //  创建日期组件 NSDateComponents *dc = [[NSDateComponents alloc]init]; //  设置当前时间 [dc setCalendar: calendar]; [dc setYear: 2014]; [dc s

IOS 多线程02-pthread 、 NSThread 、GCD 、NSOperationQueue、NSRunLoop

注:本人是翻译过来,并且加上本人的一点见解. 要点: 1.前言 2.pthread 3.NSThread 4.Grand Central Dispatch(GCD) 5.Opearation Queues 6. Run Loops 7. 多线程编程中面临的挑战 8. 资源共享 9. 互斥锁 10. 死锁 11. 资源饥饿(Starvation) 12. 优先级反转 1. 前言 其实把RunLoop叫做多线程不正确,因为它不能真正的并行,不过因为它与并发编程有莫大关系,所以值得我们深入了解. 2.

iOS多线程和NSRunLoop概述

线程概述 有些程序是一条直线,从起点到终点,如Hello World,运行打印完,它的生命周期便结束了:有些程序是一个圆,不断循环,直到将它切断,如操作系统,一直运行直到你关机.  一个运行着的程序就是一个进程或者叫做一个任务,一个进程至少包含一个线程,线程就是程序的执行流.Mac和iOS中的程序启动,创建好一个进程的同时, 一个线程便开始运行,这个线程叫主线程.主线程在程序中的地位和其他线程不同,它是其他线程最终的父线程,且所有界面的显示操作如AppKit或 UIKit的操作必须在主线程进行.

iOS多线程开发(二)---线程管理

线程管理 线程管理包括创建,配置,退出三部分.主要包括创建线程的成本,线程创建,线程属性配置,线程主体入口函数编写,线程中断等 一,线程创建成本 1,为辅助线程分配的堆栈空间大小,便于系统和进程管理,以及为函数参数和局部变量分配空间 A,内核数据结构(kernel data structures)---大约1KB,This memory is used to store the thread data structures and attributes, much of which is all

IOS多线程-个人理解

一.多线程 每一个iOS应用程序中都有一个主线程用来更新UI界面.处理用户的触摸事件.解析网络下载的数据,因此不能把一些太耗时的操作(比如网络下载数据)放在主线程中执行,不然会造成主线程堵塞(出现界面卡死,防止界面假死),带来极坏的用户体验. iOS的解决方案就是将那些耗时的操作放到另外一个线程中去执行,多线程异步编程是防止主线程堵塞,增加运行效率的最佳方法 异步:多个线程 同时执行 同步:线程排队执行并行 ->异步串行->同步 多线程技术 1.异步下载数据,是多线程技术的一个比较常见的应用场