一.通知
對(duì)于通知将硝,大家想必都不陌生恭朗,它是一個(gè)單例,允許當(dāng)事件發(fā)生時(shí)通知一些對(duì)象袋哼,讓我們?cè)诘统潭锐詈系那闆r下冀墨,來(lái)達(dá)到通信的目的。
通知的優(yōu)勢(shì):
1.不需要編寫(xiě)太多代碼涛贯,實(shí)現(xiàn)比較簡(jiǎn)單
2.對(duì)于一個(gè)發(fā)出的通知诽嘉,可以多個(gè)對(duì)象作出反應(yīng),即是說(shuō)通知是一對(duì)多的形式
通知的缺點(diǎn):
1.在編譯期不會(huì)檢查通知是否能夠被觀察者正確處理
2.在釋放注冊(cè)的對(duì)象時(shí),需要在通知中心取消注冊(cè)
3.在調(diào)試應(yīng)用時(shí)虫腋,難以跟蹤程序
4.發(fā)出通知后骄酗,不能夠從觀察者那里獲取任何反饋信息
通知的基本實(shí)現(xiàn):
- (void)viewDidLoad {
[super viewDidLoad];
[[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(test) name:@"test" object:nil];
NSLog(@"注冊(cè)通知 - %@",[NSThread currentThread]);
}
- (void)touchesBegan:(NSSet*)touches withEvent:(UIEvent *)event {
[[NSNotificationCenter defaultCenter] postNotificationName:@"test" object:nil];
NSLog(@"發(fā)送通知完成 - %@",[NSThread currentThread]);
}
- (void)test {
NSLog(@"接收到通知 - %@",[NSThread currentThread]);
sleep(3);
}
打印結(jié)果:
2017-06-13 16:53:01.040 通知的基本使用[24531:3283934] 注冊(cè)通知 -{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] 發(fā)送通知完成 -{number = 1, name = main}
注意打印結(jié)果:在test方法執(zhí)行完畢之后,才會(huì)打印發(fā)送完成的log悦冀。
如果在子線程發(fā)送通知:
- (void)viewDidLoad {? ? [super viewDidLoad];? ? [[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(test) name:@"test" object:nil]; ??
?NSLog(@"注冊(cè)通知 - %@",[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(@"發(fā)送通知完成 - %@",[NSThread currentThread]);
});
}
- (void)test {
NSLog(@"接收到通知 - %@",[NSThread currentThread]);
sleep(3);
}
打印結(jié)果:2017-06-13 17:05:01.133 通知的基本使用[25191:3296062] 注冊(cè)通知 -{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] 發(fā)送通知完成 -{number = 3, name = (null)}
得出結(jié)論:接收通知的線程和發(fā)送通知的線程是一樣的趋翻,如果在實(shí)際開(kāi)發(fā)過(guò)程中,我們是在子線程中發(fā)送通知的盒蟆,在接收到通知之后踏烙,需要刷新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(@"發(fā)送通知完成 - %@",[NSThread currentThread]);? ? ? ? });}
打印結(jié)果:2017-06-13 18:21:38.367 通知的基本使用[29365:3382047] 接收到通知 -{number = 1, name = main}
2017-06-13 18:21:41.368 通知的基本使用[29365:3382100] 發(fā)送通知完成 -{number = 3, name = (null)}
得出結(jié)論:使用NSOperationQueue可以讓接收通知的線程和發(fā)送通知的線程不一樣讨惩,讓接收通知的線程在主線程,就可以刷新UI等操作了寒屯。
二.Xcode何時(shí)會(huì)報(bào)unrecognized selector 的錯(cuò)誤
- (void)touchesBegan:(NSSet*)touches withEvent:(UIEvent *)event {
WWPerson *person = [[WWPerson alloc] init];
[person test];
}
當(dāng)向person發(fā)送test這個(gè)消息時(shí)荐捻,runtime庫(kù)會(huì)根據(jù)對(duì)象的isa指針找到該對(duì)象實(shí)際所屬的類(lèi),然后在該類(lèi)的方法列表以及父類(lèi)的方法列表里面找相應(yīng)的方法運(yùn)行寡夹,如果在最頂層的父類(lèi)中依然找不到相應(yīng)的方法實(shí)現(xiàn)時(shí)处面,程序在運(yùn)行時(shí)就會(huì)報(bào)unrecognized selector sent to的錯(cuò)誤并且崩潰,但是在此之前菩掏,objc的運(yùn)行時(shí)給出了避免程序崩潰的三次機(jī)會(huì)魂角。
1.Method resolution
objc運(yùn)行時(shí)會(huì)調(diào)用+resolveInstanceMethod:或者+resolveClassMethod:,讓我們有機(jī)會(huì)提供一個(gè)函數(shù)實(shí)現(xiàn)而不導(dǎo)致程序崩潰患蹂,如果在這里面添加了函數(shù)或颊,系統(tǒng)就會(huì)重新啟動(dòng)一次消息發(fā)送的過(guò)程,否則就會(huì)移到下一步的消息轉(zhuǎn)發(fā)传于。
+ (BOOL)resolveInstanceMethod:(SEL)sel {
if (sel == NSSelectorFromString(@"test")) {
/**
class: 給哪個(gè)類(lèi)添加方法
SEL: 添加哪個(gè)方法
IMP: 方法實(shí)現(xiàn) => 函數(shù) => 函數(shù)入口 => 函數(shù)名
type: 方法類(lèi)型:void用v來(lái)表示囱挑,id參數(shù)用@來(lái)表示,SEL用:來(lái)表示
*/
class_addMethod(self, sel, (IMP)test, "v@:@");
return YES;
}else {
return [super resolveClassMethod:sel];
}
}
void test(id self, SEL _cmd, NSNumber *meter) {
NSLog(@"測(cè)試 - WWPerson");
}
2.Fast forwarding
如果目標(biāo)對(duì)象實(shí)現(xiàn)了-forwardingTargetForSelector:的方法沼溜,runtime就會(huì)調(diào)用這個(gè)方法平挑,給我們一個(gè)機(jī)會(huì)把這個(gè)消息轉(zhuǎn)發(fā)給其他的對(duì)象,只要這個(gè)方法返回值不是nil和self系草,整個(gè)消息發(fā)送的過(guò)程就會(huì)被重啟通熄,這時(shí)發(fā)送的對(duì)象會(huì)變成我們返回的這個(gè)對(duì)象,否則就會(huì)移到下一步找都。
- (id)forwardingTargetForSelector:(SEL)aSelector {
WWTarget *target = [[WWTarget alloc] init];
if ([target respondsToSelector:aSelector]) {
return target; // 就會(huì)去調(diào)用WWTarget里面的test方法
}else {
return [super forwardingTargetForSelector: aSelector];
}
}
3.Normal Fowarding
如果上面兩種方法都沒(méi)有被實(shí)現(xiàn)的話唇辨,就會(huì)來(lái)到第三步,這是runtime給我們最后一次避免崩潰的機(jī)會(huì)能耻,首先它會(huì)-methodSignatureForSelector:來(lái)獲得函數(shù)的參數(shù)和返回值類(lèi)型赏枚,如果返回值為nil亡驰,則runtime會(huì)發(fā)出-doesNotRecognizeSelector: 的消息,程序崩潰饿幅。如果返回了一個(gè)函數(shù)簽名凡辱,runtime會(huì)創(chuàng)建一個(gè)NSInvocation對(duì)象并發(fā)送-forwardInvocation:的消息給目標(biāo)對(duì)象。
- (NSMethodSignature *)methodSignatureForSelector:(SEL)aSelector {
NSMethodSignature *signature = [NSMethodSignature signatureWithObjCTypes:"v@:"];
return signature;
}
- (void)forwardInvocation:(NSInvocation *)anInvocation {
SEL selector = [anInvocation selector];// anInvocation里面保存的是selector/target/參數(shù)
WWTarget *target = [[WWTarget alloc] init];
if ([target respondsToSelector:selector]) {
[anInvocation invokeWithTarget:target];
}
}
如果上面的三步都沒(méi)有實(shí)現(xiàn)的話栗恩,就會(huì)調(diào)用-doesNotRecognizeSelector:透乾,程序崩潰。
三.深拷貝和淺拷貝
深拷貝:內(nèi)容拷貝磕秤,拷貝出來(lái)的對(duì)象和之前的對(duì)象的地址不一樣乳乌。
淺拷貝:指針拷貝,拷貝出來(lái)的對(duì)象和之前的對(duì)象的地址一樣市咆。
直接上簡(jiǎn)單示例比較好:
1.對(duì)可變對(duì)象進(jìn)行 copy操作
- (void)viewDidLoad {
[super viewDidLoad];
NSMutableString *mStr = [NSMutableString stringWithString:@"mStr"];
NSString *copyStr = [mStr copy];
[mStr appendString:@"123"];
// mStr:0x60800007f440 - copyStr:0xa0000007274536d4
NSLog(@"mStr:%p - copyStr:%p",mStr, copyStr);
}
結(jié)論:1.對(duì)可變對(duì)象 進(jìn)行 copy 操作是內(nèi)容拷貝(深拷貝)
2. copy 出來(lái)的copyStr是NSString類(lèi)型的钦扭,如果對(duì)copyStr調(diào)用
NSMutableString的方法appendString是會(huì)崩潰的。
2.對(duì)可變對(duì)象進(jìn)行mutableCopy操作
- (void)viewDidLoad {
[super viewDidLoad];
NSMutableString *mStr = [NSMutableString stringWithString:@"mStr"];
NSMutableString *mutableCopyStr =? [mStr mutableCopy];
// str:0x608000260140 - mutableCopyStr:0x608000260440
NSLog(@"str:%p - mutableCopyStr:%p",mStr, mutableCopyStr);
}
結(jié)論:1.對(duì)可變對(duì)象 進(jìn)行 mutableCopy 操作是內(nèi)容拷貝(深拷貝)
2. mutableCopy 出來(lái)的mutableCopyStr是 NSMutableString 類(lèi)型
3.對(duì)不可變對(duì)象進(jìn)行copy操作
- (void)viewDidLoad {
[super viewDidLoad];
NSString *Str = [NSString stringWithFormat:@"Str"];
NSString *copyStr = [Str copy];
// str:0x10147e128 - copyStr:0x10147e128
NSLog(@"str:%p - copyStr:%p",Str, copyStr);
}
結(jié)論:對(duì)不可變對(duì)象 進(jìn)行 copy 操作是指針拷貝(淺拷貝)
4.對(duì)不可變對(duì)象進(jìn)行mutableCopy操作
- (void)viewDidLoad {
[super viewDidLoad];
NSString *mStr = [NSString stringWithFormat:@"mStr"];
NSMutableString *mutableCopyStr = [mStr mutableCopy];
// str:0xa0000007274536d4 - mutableCopyStr:0x60800026a680
NSLog(@"str:%p - mutableCopyStr:%p",mStr, mutableCopyStr);
}
結(jié)論:1.對(duì)不可變對(duì)象 進(jìn)行 mutableCopy操作 是內(nèi)容拷貝(深拷貝)
2.對(duì)mStr進(jìn)行mutableCopy操作的mutableCopyStr是NSMutableString類(lèi)型的
綜合以上所述:只有對(duì) 不可變對(duì)象進(jìn)行copy操作是指針拷貝(淺拷貝)床绪,其他的都是內(nèi)容拷貝(深拷貝)
四.調(diào)起鍵盤(pán)時(shí),如何將鍵盤(pán)的“換行”變成“發(fā)送/完成”等
設(shè)置returnKeyType屬性即可其弊,
UIReturnKeyDefault,
UIReturnKeyGo,// 前往
UIReturnKeyGoogle,// google
UIReturnKeyJoin,// 加入
UIReturnKeyNext,// 下一步
UIReturnKeyRoute,// 路線
UIReturnKeySearch,// 搜索
UIReturnKeySend, // 發(fā)送
UIReturnKeyYahoo,// 搜索
UIReturnKeyDone,// 完成
UIReturnKeyEmergencyCall,// 緊急電話
UIReturnKeyContinue NS_ENUM_AVAILABLE_IOS(9_0),// 繼續(xù)
五.viewDidLayoutSubviews和layoutSubviews的調(diào)用順序
viewDidLayoutSubviews在 layoutSubviews前面調(diào)用
layoutSubviews在drawRect :前面調(diào)用
2017-06-14 10:31:35.215 layoutSubviews等的調(diào)用順序[7357:98975] -[ViewController viewDidLoad]
2017-06-14 10:31:35.215 layoutSubviews等的調(diào)用順序[7357:98975] -[WWView initWithFrame:]
2017-06-14 10:31:35.220 layoutSubviews等的調(diào)用順序[7357:98975] -[ViewController viewWillLayoutSubviews]
2017-06-14 10:31:35.220 layoutSubviews等的調(diào)用順序[7357:98975] -[ViewController viewDidLayoutSubviews]
2017-06-14 10:31:35.220 layoutSubviews等的調(diào)用順序[7357:98975] -[WWView layoutSubviews]
2017-06-14 10:31:35.221 layoutSubviews等的調(diào)用順序[7357:98975] -[WWView drawRect:]
六.如何給分類(lèi)動(dòng)態(tài)添加屬性#import "WWView+Tools.h"#importstatic char strKey;
@implementation WWView (Tools)
- (void)setDynamicStr:(NSString *)dynamicStr {
/**
id object: 需要給哪個(gè)對(duì)象的屬性賦值
const void *key:屬性對(duì)應(yīng)的key值
id value:設(shè)置屬性的值為value
objc_AssociationPolicy policy:關(guān)聯(lián)策略 枚舉值 一般選擇NONATOMIC
*/
objc_setAssociatedObject(self, &strKey, dynamicStr, OBJC_ASSOCIATION_COPY_NONATOMIC);
}
- (NSString *)dynamicStr {
return objc_getAssociatedObject(self, &strKey);
}
七.如何把一個(gè)view生成一張圖片癞己,并且保存到本地因?yàn)樯婕暗皆L問(wèn)相冊(cè),所以先在plist文件里面添加NSPhotoLibraryUsageDescription允許應(yīng)用程序訪問(wèn)你的相冊(cè)
- (void)touchesBegan:(NSSet*)touches withEvent:(UIEvent *)event {
dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{
// self.testView:要生成為圖片的view
UIGraphicsBeginImageContextWithOptions(self.testView.bounds.size, 0, [[UIScreen mainScreen] scale]);
[self.testView.layer renderInContext:UIGraphicsGetCurrentContext()];
UIImage *viewImage = UIGraphicsGetImageFromCurrentImageContext();
UIGraphicsEndImageContext();
UIImageWriteToSavedPhotosAlbum(viewImage, self, @selector(imageSavedToPhotosAlbum:didFinishSavingWithError:contextInfo:), nil);
});
}
- (void)imageSavedToPhotosAlbum:(UIImage *)image didFinishSavingWithError:(NSError *)error contextInfo:(void *)contextInfo
{
if (!error) {
NSLog(@"成功");
}else {
NSLog(@"失敗 - %@",error);
}
}
八.如果服務(wù)器返回給我們的數(shù)據(jù)是包含標(biāo)簽的梭伐,我們應(yīng)該如何加載
// html_content:含有html標(biāo)簽的富文本
1.用UILabel去加載
NSMutableAttributedString *attributeStr = [[NSMutableAttributedString alloc] initWithData:[html_content dataUsingEncoding:NSUnicodeStringEncoding] options:@{NSDocumentTypeDocumentAttribute:NSHTMLTextDocumentType? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? }documentAttributes:nil error:nil];
self.contentLabel.attributedText = attributeStr;
PS:如果要改變文本的字體大小顏色等痹雅,一定要在這后面改
2.直接使用UIWebView去加載
[self.contentWebView loadHTMLString:html_content baseURL:nil];
九.上傳到應(yīng)用商店太慢的話,怎么解決
可以考慮使用Xcode - Open Developer Tool - Application Loader來(lái)解決