iOS通知

2017-06-23  本文已影响0人  Persistence__

一.通知

对于通知,大家想必都不陌生,它是一个单例,允许当事件发生时通知一些对象,让我们在低程度耦合的情况下,来达到通信的目的。

通知的优势:

1.不需要编写太多代码,实现比较简单

2.对于一个发出的通知,可以多个对象作出反应,即是说通知是一对多的形式

通知的缺点:

1.在编译期不会检查通知是否能够被观察者正确处理

2.在释放注册的对象时,需要在通知中心取消注册

3.在调试应用时,难以跟踪程序

4.发出通知后,不能够从观察者那里获取任何反馈信息

通知的基本实现:

- (void)viewDidLoad {

[super viewDidLoad];

[[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(test) name:@"test" object:nil];

NSLog(@"注册通知 - %@",[NSThread currentThread]);

}

- (void)touchesBegan:(NSSet *)touches withEvent:(UIEvent *)event {

[[NSNotificationCenter defaultCenter] postNotificationName:@"test" object:nil];

NSLog(@"发送通知完成 - %@",[NSThread currentThread]);

}

- (void)test {

NSLog(@"接收到通知 - %@",[NSThread currentThread]);

sleep(3);

}

打印结果:

2017-06-13 16:53:01.040 通知的基本使用[24531:3283934] 注册通知 - {number = 1, name = main}

2017-06-13 16:53:10.334 通知的基本使用[24531:3283934] 接收到通知 - {number = 1, name = main}

2017-06-13 16:53:13.335 通知的基本使用[24531:3283934] 发送通知完成 - {number = 1, name = main}

注意打印结果:在test方法执行完毕之后,才会打印发送完成的log。

如果在子线程发送通知:

- (void)viewDidLoad {

[super viewDidLoad];

[[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(test) name:@"test" object:nil];

NSLog(@"注册通知 - %@",[NSThread currentThread]);

}

- (void)touchesBegan:(NSSet *)touches withEvent:(UIEvent *)event {

dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_BACKGROUND, 0), ^{

NSNotification *notification = [NSNotification notificationWithName:@"test"

object:nil];

// NSPostASAP是接收不到通知的 要使用NSPostNow

[[NSNotificationQueue defaultQueue] enqueueNotification:notification postingStyle:NSPostNow];

NSLog(@"发送通知完成 - %@",[NSThread currentThread]);

});

}

- (void)test {

NSLog(@"接收到通知 - %@",[NSThread currentThread]);

sleep(3);

}

打印结果:

2017-06-13 17:05:01.133 通知的基本使用[25191:3296062] 注册通知 - {number = 1, name = main}

2017-06-13 17:05:02.423 通知的基本使用[25191:3296125] 接收到通知 - {number = 3, name = (null)}

2017-06-13 17:05:05.523 通知的基本使用[25191:3296125] 发送通知完成 - {number = 3, name = (null)}

得出结论:接收通知的线程和发送通知的线程是一样的,如果在实际开发过程中,我们是在子线程中发送通知的,在接收到通知之后,需要刷新UI等操作,一定要回到主线程。

- (void)viewDidLoad {

[super viewDidLoad];

_observe = [[NSNotificationCenter defaultCenter] addObserverForName:@"test" object:nil queue:[NSOperationQueue mainQueue] usingBlock:^(NSNotification * _Nonnull note) {

NSLog(@"接收到通知 - %@",[NSThread currentThread]);

sleep(3);

}];

}

- (void)touchesBegan:(NSSet *)touches withEvent:(UIEvent *)event {

dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_BACKGROUND, 0), ^{

[[NSNotificationCenter defaultCenter] postNotificationName:@"test" object:nil];

NSLog(@"发送通知完成 - %@",[NSThread currentThread]);

});

}

打印结果:

2017-06-13 18:21:38.367 通知的基本使用[29365:3382047] 接收到通知 - {number = 1, name = main}

2017-06-13 18:21:41.368 通知的基本使用[29365:3382100] 发送通知完成 - {number = 3, name = (null)}

得出结论:使用NSOperationQueue可以让接收通知的线程和发送通知的线程不一样,让接收通知的线程在主线程,就可以刷新UI等操作了。

上一篇下一篇

猜你喜欢

热点阅读