1:释放单例的方法
static dispatch_once_t onceToken;
onceToke = 0;
2:NSNotificationCenter 在哪个线程post则在哪个线程转发,不是add时候的线程
主线程执行
[[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(noti) name:@"abc" object:nil];
dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{
//打印[NSThread currentThread]
[[NSNotificationCenter defaultCenter] postNotificationName:@"abc" object:@"a"];
});
- (void)noti{
//打印[NSThread currentThread]
NSLog(@"noti");
}
结果:
{number = 3, name = (null)}
{number = 3, name = (null)}
如果noti方法是要更新UI,请确保post在主线程上或者在noti方法内处理一下。
3:注册通讯录变化通知
ABAddressBookRegisterExternalChangeCallback(addressBook, addressBookChanged2, nil);
只能注册一次,否则当通讯录改变时,注册多少次就会调用多少次addressBookChanged2
4:NSTimer
invalidate方法调用必须在timer添加到的runloop所在的线程
http://www.cnblogs.com/mddblog/p/6517377.html
5:等待两个请求完成之后再更新UI
方法一:http://www.jianshu.com/p/943dcb9ad632
方法二:http://www.jianshu.com/p/228403206664
dispatch_group_enter(dispatchGroup);
dispatch_group_leave(dispatchGroup);
dispatch_queue_t queue = dispatch_queue_create("getHomePageQueue", DISPATCH_QUEUE_SERIAL);
dispatch_async(queue, ^{
dispatch_group_t group = dispatch_group_create();
dispatch_group_enter(group);
[self.dataRequestUC requestHomePageDataWithParama:@{} finishBlock:^(id requestResult,NSInteger statusCode) {
if (requestResult) {
__strong typeof(weakSelf) strongSelf = weakSelf;
NSMutableArray *home = [NSMutableArray arrayWithArray:[strongSelf corprationCMSData:requestResult]];
strongSelf.homePageData = [home mutableCopy];
dispatch_group_leave(group);
}
}];
dispatch_group_enter(group);
[self.dataRequestUC requestGJMessageWithParama:@{@"pageSize":@"3",@"pageNo":@"1"} finishBlock:^(id requestResult, NSInteger statusCode) {
__strong typeof(weakSelf) strongSelf = weakSelf;
if ([requestResult isKindOfClass:[NSDictionary class]]) {
NSDictionary *body = requestResult[@"body"];
NSArray *data = body[@"data"];
[strongSelf corperationMessage:data];
dispatch_group_leave(group);
}
}];
long timeout = dispatch_group_wait(group, dispatch_time(DISPATCH_TIME_NOW, (int64_t)10*NSEC_PER_SEC));
if (timeout) {
dataBlock(nil);
}
dispatch_group_notify(group, dispatch_get_main_queue(), ^{
dataBlock([self combineData]);
});
dispatch_group_wait不能在主线程使用,否则会阻塞。
方法三:RAC
方法四:dispatch_barrier_async
方法五:NSOperationQueue addDependency
6:自定义NSOperation
重写start方法支持异步操作,但是需要管理状态,并且以KVO的方式进行实现
重写main方法不用管理operation的状态,为了能够使用操作队列的取消功能,需要不断的检查isCancel
- (void)main
{
while (notDone && !self.isCancelled) {
// 进行处理
}
}
7:在 Objective-C 中将属性以 atomic 的形式来声明,就能支持互斥锁了。默认就为atomic,但是开销较大。
加锁解锁有性能消耗,当获取锁的时候,如果属性已经某一个线程操作,则当前线程需要等待,等属性被操作完毕,解锁后才能继续进行。
8:如果你在 table view 或者是 collection view 的 cell 上做了自定义绘制的话,最好将它们放入 operation 的子类中去。你可以将它们添加到后台操作队列,也可以在用户将 cell 滚动出边界时的 didEndDisplayingCell 委托方法中进行取消
9:在后台或者子线程访问UIKit/APPKit等属性,如设置image则设置生效的时间会延后。如果有两个及以上的线程同时进行访问或设置,会造成时机性的崩溃问题。
10:iOS中用来表示内存的堆栈与数据结构中的堆栈代表的意义不一样
栈:先进后出
全局和静态变量在堆中
指针放在栈中,APP通过在栈中找到数据的指针,指向堆中的数据。
11:创建静态库,使用cocoapod管理第三方依赖。pod install以后会自动在link Binary With Libraries中将工程名的静态库放进去,导致编译报错
error: /Applications/Xcode.app/Contents/Developer/Toolchains/XcodeDefault.xctoolchain/usr/bin/libtool: can't locate file for: -lPods-projectName
error: /Applications/Xcode.app/Contents/Developer/Toolchains/XcodeDefault.xctoolchain/usr/bin/libtool: file: -lPods-projectName is not an object file (not allowed in a library)
手动删除后,编译成功。