iOS里關(guān)于block的一些理解

介紹

block實(shí)際上就是Objective-C語(yǔ)言對(duì)于閉包的實(shí)現(xiàn)锤窑。
block配合上dispatch_queue璧针,可以方便地實(shí)現(xiàn)簡(jiǎn)單的多線程編程和異步編程。
(閉包是一個(gè)函數(shù)(或指向函數(shù)的指針)渊啰,再加上該函數(shù)執(zhí)行的外部的上下文變量(有時(shí)候也稱(chēng)作自由變量)探橱。)

block的寫(xiě)法

回傳值(^名字)(參數(shù)列);

//聲明一個(gè)square的Block Pointer,其所指向的Block有一個(gè)int輸入和int輸出  
int (^square)(int);  
//將Block實(shí)體指定給square  
square = ^(int a){ return a*a ; };  
//調(diào)用方法绘证,感覺(jué)是是不是很像function的用法隧膏?  
int result = square(5);  
NSLog(@"%d", result);  

當(dāng)其作為Object-C method的傳入值的話(huà),需要把類(lèi)型寫(xiě)在變量前面嚷那,然后加上小括號(hào)胞枕。比如下面這種寫(xiě)法:

//square參數(shù)的類(lèi)型是int(^)(int)  
-(void)objcMethod:(int(^)(int))square;  

block陣列的使用

{
    void (^blocks[3])(void);
    for (NSInteger i = 0; i < 3; i++) {
        blocks[i] = ^{
            NSLog(@"Hello:%i", i);
        };
    }
    blocks[0](); //result:Hello:0
    blocks[1](); //result:Hello:1
    blocks[2](); //result:Hello:2
}

存取變量

Block將使用到的、作用域附近的變量的值建立一份快照拷貝到棧上车酣。

1曲稼、讀取和Block pointer同一個(gè)Scope的變量值:

{  
    int outA = 8;  
    int (^myPtr)(int) = ^(int a){ return outA + a;};  
    //block里面可以讀取同一類(lèi)型的outA的值  
    int result = myPtr(3);  // result is 11  
    NSLog(@"result=%d", result);  
}  

下面這一段代碼就不一樣了

{
    int outA = 8;  
    int (^myPtr)(int) = ^(int a){ return outA + a;}; //block里面可以讀取同一類(lèi)型的outA的值  
    outA = 5;  //在調(diào)用myPtr之前改變outA的值  
    int result = myPtr(3);  // result的值仍然是11,并不是8  
    NSLog(@"result=%d", result);  
}  

為什么result 的值仍然是11湖员?而不是8呢贫悄?事實(shí)上,myPtr在其主體中用到的outA這個(gè)變量值的時(shí)候做了一個(gè)copy的動(dòng)作娘摔,把outA的值copy下來(lái)窄坦,在Block中作為常量使用。所以凳寺,之后outA即使換成了新的值鸭津,對(duì)于myPtr里面copy的值是沒(méi)有影響的。(類(lèi)似于深拷貝)

需要注意的是肠缨,這里copy的值是變量的值逆趋,如果它是一個(gè)記憶體的位置(地址),換句話(huà)說(shuō)晒奕,就是這個(gè)變量是個(gè)指針的話(huà)闻书,它的值是可以在block里被改變的。(相當(dāng)于淺拷貝脑慧,拷貝的只是一個(gè)指針地址魄眉,對(duì)象地址還是沒(méi)變的)

{  
    NSMutableArray \*mutableArray = [NSMutableArray arrayWithObjects:@"one", @"two", @"three", nil];  
    int result = ^(int a){[mutableArray removeLastObject]; return a*a;}(5);  
    NSLog(@"test array :%@", mutableArray);  
}  
    
//原本mutableArray的值是{@"one",@"two",@"three"},在block里面被更改mutableArray后闷袒,就變成{@"one", @"two"}了坑律。

2、直接存取static類(lèi)型的變量

因?yàn)槿肿兞炕蜢o態(tài)變量在內(nèi)存中的地址是固定的囊骤,Block在讀取該變量值的時(shí)候是直接從其所在內(nèi)存讀出晃择,獲取到的是最新值,而不是在定義時(shí)copy的常量淘捡。

{  
    static int outA = 8;  
    int (^myPtr)(int) = ^(int a){return outA + a;};  
    outA = 5;  
    int result = myPtr(3);  
    //result的值是8藕各,因?yàn)閛utA是static類(lèi)型的變量 (該變量在全局?jǐn)?shù)據(jù)區(qū)分配內(nèi)存,但作用域還是局部作用域) 
    NSLog(@"result=%d", result);     
}  

3、Block Variable類(lèi)型的變量

在某個(gè)變量前面如果加上修飾字“__block”的話(huà)(注意焦除,block前面有兩個(gè)下劃線)激况,這個(gè)變量就稱(chēng)作block variable”炱牵基本類(lèi)型的Block變量等效于全局變量乌逐、或靜態(tài)變量。

那么在block里面就可以任意修改此變量的值创葡,如下代碼:

{  
    __block int num = 5;  
      
    int (^myPtr)(int) = ^(int a){return num++;};  
    int (^myPtr2)(int) = ^(int a){return num++;};  
    int result = myPtr(0);   //result的值為5浙踢,num的值為6  
    result = myPtr2(0);      //result的值為6,num的值為7  
    NSLog(@"result=%d", result);      
}

4灿渴、weak–strong dance(避免循環(huán)引用)

  • 使用方將self或成員變量加入block之前要先將self變?yōu)開(kāi)_weak
  • 在多線程環(huán)境下(block中的weakSelf有可能被析構(gòu)的情況下)洛波,需要先將self轉(zhuǎn)為strong指針胰舆,避免在運(yùn)行到某個(gè)關(guān)鍵步驟時(shí)self對(duì)象被析構(gòu)。

以上兩條合起來(lái)有個(gè)名詞叫weak–strong dance

以下是使用weak–strong dance的經(jīng)典代碼

__weak __typeof(self)weakSelf = self和
__strong __typeof(weakSelf)strongSelf = weakSelf

//AFNetworking經(jīng)典代碼
__weak __typeof(self)weakSelf = self;
AFNetworkReachabilityStatusBlock callback = ^(AFNetworkReachabilityStatus status) {
    __strong __typeof(weakSelf)strongSelf = weakSelf;
    strongSelf.networkReachabilityStatus = status;
    if (strongSelf.networkReachabilityStatusBlock) {
        strongSelf.networkReachabilityStatusBlock(status);
    }
};

其中用到了__typeof(self)蹬挤,這里涉及幾個(gè)知識(shí)點(diǎn):

a. __typeof缚窿、__typeof__、typeof的區(qū)別
恩~~他們沒(méi)有區(qū)別焰扳,但是這牽扯一段往事倦零,在早期C語(yǔ)言中沒(méi)有typeof這個(gè)關(guān)鍵字,__typeof吨悍、__typeof__是在C語(yǔ)言的擴(kuò)展關(guān)鍵字的時(shí)候出現(xiàn)的扫茅。
typeof是現(xiàn)代GNU C++的關(guān)鍵字,從Objective-C的根源說(shuō)育瓜,他其實(shí)來(lái)自于C語(yǔ)言葫隙,所以AFNetworking使用了繼承自C的關(guān)鍵字。

b.對(duì)于老的LLVM編譯器上面這句話(huà)會(huì)編譯報(bào)錯(cuò)躏仇,所以在很早的ARC使用者中流行__typeof(&*self)這種寫(xiě)法停蕉,原因如下
大致說(shuō)法是老LLVM編譯器會(huì)將__typeof轉(zhuǎn)義為 XXX類(lèi)名 const __strong的__strong和前面的__weak關(guān)鍵字對(duì)指針的修飾又沖突了,所以加上&對(duì)指針的修飾钙态。

第四慧起、五、六行册倒,如果不轉(zhuǎn)成strongSelf而使用weakSelf蚓挤,后面幾句話(huà)中,有可能在第四句執(zhí)行之后self的對(duì)象可能被析構(gòu)掉驻子,然后后面的StausBlock沒(méi)有執(zhí)行灿意,導(dǎo)致邏輯錯(cuò)誤。

最后第五行崇呵,使用前對(duì)block判空缤剧。

 //以下代碼是對(duì)__weak __typeof(self)weakSelf = self
 //和__strong __typeof(weakSelf)strongSelf = weakSelf的宏定義
#ifndef weakify
    #if DEBUG
        #if __has_feature(objc_arc)
        #define weakify(object) autoreleasepool{} __weak __typeof__(object) weak##_##object = object;
        #else
        #define weakify(object) autoreleasepool{} __block __typeof__(object) block##_##object = object;
        #endif
    #else
        #if __has_feature(objc_arc)
        #define weakify(object) try{} @finally{} {} __weak __typeof__(object) weak##_##object = object;
        #else
        #define weakify(object) try{} @finally{} {} __block __typeof__(object) block##_##object = object;
        #endif
    #endif
#endif

#ifndef strongify
    #if DEBUG
        #if __has_feature(objc_arc)
        #define strongify(object) autoreleasepool{} __typeof__(object) object = weak##_##object;
        #else
        #define strongify(object) autoreleasepool{} __typeof__(object) object = block##_##object;
        #endif
    #else
        #if __has_feature(objc_arc)
        #define strongify(object) try{} @finally{} __typeof__(object) object = weak##_##object;
        #else
        #define strongify(object) try{} @finally{} __typeof__(object) object = block##_##object;
        #endif
    #endif
#endif
//使用方法
@weakify(self);
AFNetworkReachabilityStatusBlock callback = ^(AFNetworkReachabilityStatus status) {
    @strongify(self)
    if(!self)return; 
    self.networkReachabilityStatus = status; 
    if (self.networkReachabilityStatusBlock) {
        self.networkReachabilityStatusBlock(status);
    }
};

避免循環(huán)引用

為什么會(huì)發(fā)生循環(huán)引用呢?

因?yàn)閷?duì)象obj在Block被copy到堆上的時(shí)候自動(dòng)retain了一次域慷。因?yàn)锽lock不知道obj什么時(shí)候被釋放荒辕,為了不在Block使用obj前被釋放,Block retain了obj一次犹褒,在Block被釋放的時(shí)候抵窒,obj被release一次。

retain cycle問(wèn)題的根源在于Block和obj可能會(huì)互相強(qiáng)引用叠骑,互相retain對(duì)方李皇,這樣就導(dǎo)致了retain cycle,最后這個(gè)Block和obj就變成了孤島宙枷,誰(shuí)也釋放不了誰(shuí)掉房。

會(huì)發(fā)生循環(huán)引用例子的demo

參考:

http://www.cnblogs.com/zhangyang17/p/4667621.html

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末茧跋,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子卓囚,更是在濱河造成了極大的恐慌厌衔,老刑警劉巖,帶你破解...
    沈念sama閱讀 218,036評(píng)論 6 506
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件捍岳,死亡現(xiàn)場(chǎng)離奇詭異,居然都是意外死亡睬隶,警方通過(guò)查閱死者的電腦和手機(jī)锣夹,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,046評(píng)論 3 395
  • 文/潘曉璐 我一進(jìn)店門(mén),熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái)苏潜,“玉大人银萍,你說(shuō)我怎么就攤上這事⌒糇螅” “怎么了贴唇?”我有些...
    開(kāi)封第一講書(shū)人閱讀 164,411評(píng)論 0 354
  • 文/不壞的土叔 我叫張陵,是天一觀的道長(zhǎng)飞袋。 經(jīng)常有香客問(wèn)我戳气,道長(zhǎng),這世上最難降的妖魔是什么巧鸭? 我笑而不...
    開(kāi)封第一講書(shū)人閱讀 58,622評(píng)論 1 293
  • 正文 為了忘掉前任瓶您,我火速辦了婚禮,結(jié)果婚禮上纲仍,老公的妹妹穿的比我還像新娘呀袱。我一直安慰自己,他們只是感情好郑叠,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,661評(píng)論 6 392
  • 文/花漫 我一把揭開(kāi)白布。 她就那樣靜靜地躺著,像睡著了一般攒菠。 火紅的嫁衣襯著肌膚如雪亚情。 梳的紋絲不亂的頭發(fā)上,一...
    開(kāi)封第一講書(shū)人閱讀 51,521評(píng)論 1 304
  • 那天沸版,我揣著相機(jī)與錄音婉宰,去河邊找鬼。 笑死推穷,一個(gè)胖子當(dāng)著我的面吹牛心包,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播馒铃,決...
    沈念sama閱讀 40,288評(píng)論 3 418
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼蟹腾,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼痕惋!你這毒婦竟也來(lái)了?” 一聲冷哼從身側(cè)響起娃殖,我...
    開(kāi)封第一講書(shū)人閱讀 39,200評(píng)論 0 276
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤值戳,失蹤者是張志新(化名)和其女友劉穎,沒(méi)想到半個(gè)月后炉爆,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體堕虹,經(jīng)...
    沈念sama閱讀 45,644評(píng)論 1 314
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,837評(píng)論 3 336
  • 正文 我和宋清朗相戀三年芬首,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了赴捞。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 39,953評(píng)論 1 348
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡郁稍,死狀恐怖赦政,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情耀怜,我是刑警寧澤恢着,帶...
    沈念sama閱讀 35,673評(píng)論 5 346
  • 正文 年R本政府宣布,位于F島的核電站财破,受9級(jí)特大地震影響掰派,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜左痢,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,281評(píng)論 3 329
  • 文/蒙蒙 一碗淌、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧抖锥,春花似錦亿眠、人聲如沸。這莊子的主人今日做“春日...
    開(kāi)封第一講書(shū)人閱讀 31,889評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)。三九已至拯勉,卻和暖如春竟趾,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背宫峦。 一陣腳步聲響...
    開(kāi)封第一講書(shū)人閱讀 33,011評(píng)論 1 269
  • 我被黑心中介騙來(lái)泰國(guó)打工岔帽, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人导绷。 一個(gè)月前我還...
    沈念sama閱讀 48,119評(píng)論 3 370
  • 正文 我出身青樓犀勒,卻偏偏與公主長(zhǎng)得像,于是被迫代替她去往敵國(guó)和親。 傳聞我的和親對(duì)象是個(gè)殘疾皇子贾费,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,901評(píng)論 2 355

推薦閱讀更多精彩內(nèi)容

  • 前言 Blocks是C語(yǔ)言的擴(kuò)充功能钦购,而Apple 在OS X Snow Leopard 和 iOS 4中引入了這...
    小人不才閱讀 3,768評(píng)論 0 23
  • 《Objective-C高級(jí)編程》這本書(shū)就講了三個(gè)東西:自動(dòng)引用計(jì)數(shù)、block褂萧、GCD押桃,偏向于從原理上對(duì)這些內(nèi)容...
    WeiHing閱讀 9,811評(píng)論 10 69
  • 人都喜歡把心 放在柔軟舒適的地方 可惜卻未必安全。
    不知不覺(jué)云閱讀 129評(píng)論 0 1
  • 每天匆匆忙忙上班磕昼,華燈初夏才回到家,地鐵是我最常用的交通工具舶得。每天在地鐵里各種聲音充斥著耳膜。那天爽蝴,兩位三十多歲的...
    風(fēng)中尋夢(mèng)閱讀 1,632評(píng)論 84 58
  • 我只是想用安東尼的方式來(lái)記錄生活沐批,編寫(xiě)自己的故事。 上午從家回到學(xué)校蝎亚,因?yàn)樽蛲淼囊灰篃o(wú)眠九孩,一路上昏昏沉沉地睡了過(guò)來(lái)...
    夏懮閱讀 163評(píng)論 0 0