一.通知
對于通知,大家想必都不陌生,它是一個單例,允許當事件發生時通知一些對象,讓我們在低程度耦合的情況下,來達到通信的目的。
通知的優勢:
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<UITouch *> *)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] 注冊通知 - <NSThread: 0x600000079c80>{number = 1, name = main}
2017-06-13 16:53:10.334 通知的基本使用[24531:3283934] 接收到通知 - <NSThread: 0x600000079c80>{number = 1, name = main}
2017-06-13 16:53:13.335 通知的基本使用[24531:3283934] 發送通知完成 - <NSThread: 0x600000079c80>{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<UITouch *> *)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] 注冊通知 - <NSThread: 0x608000076440>{number = 1, name = main}
2017-06-13 17:05:02.423 通知的基本使用[25191:3296125] 接收到通知 - <NSThread: 0x608000267980>{number = 3, name = (null)}
2017-06-13 17:05:05.523 通知的基本使用[25191:3296125] 發送通知完成 - <NSThread: 0x608000267980>{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<UITouch *> *)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] 接收到通知 - <NSThread: 0x600000063d80>{number = 1, name = main}
2017-06-13 18:21:41.368 通知的基本使用[29365:3382100] 發送通知完成 - <NSThread: 0x600000071bc0>{number = 3, name = (null)}
得出結論:使用NSOperationQueue
可以讓接收通知的線程和發送通知的線程不一樣,讓接收通知的線程在主線程,就可以刷新UI等操作了。
二.Xcode何時會報unrecognized selector
的錯誤
- (void)touchesBegan:(NSSet<UITouch *> *)touches withEvent:(UIEvent *)event {
WWPerson *person = [[WWPerson alloc] init];
[person test];
}
當向person
發送test
這個消息時,runtime
庫會根據對象的isa
指針找到該對象實際所屬的類,然后在該類的方法列表以及父類的方法列表里面找相應的方法運行,如果在最頂層的父類中依然找不到相應的方法實現時,程序在運行時就會報unrecognized selector sent to
的錯誤并且崩潰,但是在此之前,objc的運行時給出了避免程序崩潰的三次機會。
- Method resolution
objc運行時會調用+resolveInstanceMethod:
或者+resolveClassMethod:
,讓我們有機會提供一個函數實現而不導致程序崩潰,如果在這里面添加了函數,系統就會重新啟動一次消息發送的過程,否則就會移到下一步的消息轉發。
+ (BOOL)resolveInstanceMethod:(SEL)sel {
if (sel == NSSelectorFromString(@"test")) {
/**
class: 給哪個類添加方法
SEL: 添加哪個方法
IMP: 方法實現 => 函數 => 函數入口 => 函數名
type: 方法類型:void用v來表示,id參數用@來表示,SEL用:來表示
*/
class_addMethod(self, sel, (IMP)test, "v@:@");
return YES;
}else {
return [super resolveClassMethod:sel];
}
}
void test(id self, SEL _cmd, NSNumber *meter) {
NSLog(@"測試 - WWPerson");
}
2.Fast forwarding
如果目標對象實現了-forwardingTargetForSelector:
的方法,runtime就會調用這個方法,給我們一個機會把這個消息轉發給其他的對象,只要這個方法返回值不是nil
和self
,整個消息發送的過程就會被重啟,這時發送的對象會變成我們返回的這個對象,否則就會移到下一步。
- (id)forwardingTargetForSelector:(SEL)aSelector {
WWTarget *target = [[WWTarget alloc] init];
if ([target respondsToSelector:aSelector]) {
return target; // 就會去調用WWTarget里面的test方法
}else {
return [super forwardingTargetForSelector: aSelector];
}
}
3.Normal Fowarding
如果上面兩種方法都沒有被實現的話,就會來到第三步,這是runtime給我們最后一次避免崩潰的機會,首先它會-methodSignatureForSelector:
來獲得函數的參數和返回值類型,如果返回值為nil
,則runtime會發出-doesNotRecognizeSelector:
的消息,程序崩潰。如果返回了一個函數簽名,runtime會創建一個NSInvocation對象并發送-forwardInvocation:
的消息給目標對象。
- (NSMethodSignature *)methodSignatureForSelector:(SEL)aSelector {
NSMethodSignature *signature = [NSMethodSignature signatureWithObjCTypes:"v@:"];
return signature;
}
- (void)forwardInvocation:(NSInvocation *)anInvocation {
SEL selector = [anInvocation selector];// anInvocation里面保存的是selector/target/參數
WWTarget *target = [[WWTarget alloc] init];
if ([target respondsToSelector:selector]) {
[anInvocation invokeWithTarget:target];
}
}
如果上面的三步都沒有實現的話,就會調用-doesNotRecognizeSelector:
,程序崩潰。
三.深拷貝和淺拷貝
深拷貝:內容拷貝,拷貝出來的對象和之前的對象的地址不一樣。
淺拷貝:指針拷貝,拷貝出來的對象和之前的對象的地址一樣。
直接上簡單示例比較好:
1.對可變對象進行 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);
}
結論:1.對可變對象 進行 copy 操作是內容拷貝(深拷貝)
2. copy 出來的copyStr是NSString類型的,如果對copyStr調用
NSMutableString的方法appendString是會崩潰的。
2.對可變對象進行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);
}
結論:1.對可變對象 進行 mutableCopy 操作是內容拷貝(深拷貝)
2. mutableCopy 出來的mutableCopyStr是 NSMutableString 類型
3.對不可變對象進行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);
}
結論:對不可變對象 進行 copy 操作是指針拷貝(淺拷貝)
4.對不可變對象進行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);
}
結論:1.對不可變對象 進行 mutableCopy操作 是內容拷貝(深拷貝)
2.對mStr進行mutableCopy操作的mutableCopyStr是NSMutableString類型的
綜合以上所述:只有對 不可變對象
進行copy
操作是指針拷貝(淺拷貝),其他的都是內容拷貝(深拷貝)
四.調起鍵盤時,如何將鍵盤的“換行”變成“發送/完成”等
設置returnKeyType
屬性即可,
UIReturnKeyDefault,
UIReturnKeyGo,// 前往
UIReturnKeyGoogle,// google
UIReturnKeyJoin,// 加入
UIReturnKeyNext,// 下一步
UIReturnKeyRoute,// 路線
UIReturnKeySearch,// 搜索
UIReturnKeySend, // 發送
UIReturnKeyYahoo,// 搜索
UIReturnKeyDone,// 完成
UIReturnKeyEmergencyCall,// 緊急電話
UIReturnKeyContinue NS_ENUM_AVAILABLE_IOS(9_0),// 繼續
五.viewDidLayoutSubviews和layoutSubviews的調用順序
viewDidLayoutSubviews
在 layoutSubviews
前面調用
layoutSubviews
在drawRect :
前面調用
2017-06-14 10:31:35.215 layoutSubviews等的調用順序[7357:98975] -[ViewController viewDidLoad]
2017-06-14 10:31:35.215 layoutSubviews等的調用順序[7357:98975] -[WWView initWithFrame:]
2017-06-14 10:31:35.220 layoutSubviews等的調用順序[7357:98975] -[ViewController viewWillLayoutSubviews]
2017-06-14 10:31:35.220 layoutSubviews等的調用順序[7357:98975] -[ViewController viewDidLayoutSubviews]
2017-06-14 10:31:35.220 layoutSubviews等的調用順序[7357:98975] -[WWView layoutSubviews]
2017-06-14 10:31:35.221 layoutSubviews等的調用順序[7357:98975] -[WWView drawRect:]
六.如何給分類動態添加屬性
#import "WWView+Tools.h"
#import <objc/runtime.h>
static char strKey;
@implementation WWView (Tools)
- (void)setDynamicStr:(NSString *)dynamicStr {
/**
id object: 需要給哪個對象的屬性賦值
const void *key:屬性對應的key值
id value:設置屬性的值為value
objc_AssociationPolicy policy:關聯策略 枚舉值 一般選擇NONATOMIC
*/
objc_setAssociatedObject(self, &strKey, dynamicStr, OBJC_ASSOCIATION_COPY_NONATOMIC);
}
- (NSString *)dynamicStr {
return objc_getAssociatedObject(self, &strKey);
}
七.如何把一個view生成一張圖片,并且保存到本地
因為涉及到訪問相冊,所以先在plist文件里面添加NSPhotoLibraryUsageDescription
允許應用程序訪問你的相冊
- (void)touchesBegan:(NSSet<UITouch *> *)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);
}
}
八.如果服務器返回給我們的數據是包含標簽的,我們應該如何加載
// html_content:含有html標簽的富文本
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去加載
//設置字體大小為15,顏色rgb(124,181,236),邊距為15,并且圖片的寬度自動充滿屏幕,高度自適應
NSString *html_content = @"要加載的html內容";
NSString *htmls = [NSString stringWithFormat:@"<html> \n"
"<head> \n"
"<style type=\"text/css\"> \n"
"body {margin:15;font-size:15;color:%@}\n"
"</style> \n"
"</head> \n"
"<body>"
"<script type='text/javascript'>"
"window.onload = function(){\n"
"var $img = document.getElementsByTagName('img');\n"
"for(var p in $img){\n"
" $img[p].style.width = '100%%';\n"
"$img[p].style.height ='auto'\n"
"}\n"
"}"
"</script>%@"
"</body>"
"</html>",@"rgb(124,181,236)", html_content];
[self.contentWebView loadHTMLString:htmls baseURL:nil];
九.上傳到應用商店太慢的話,怎么解決
可以考慮使用Xcode - Open Developer Tool - Application Loader來解決
十.利用Application Loader打包提交到App Store可能會遇到錯誤:
The filename 未命名.ipa in the package contains an invalid character(s). The valid characters are:A-Z
,a-z,0-9,dash,period,underscore,but the name cannot start with a dash,period,or underscore.
解決辦法:在Archive之后的包不能再試中文名,把XXX.ipa改成英文名就搞定了。