Objective-C底層探究之block(二)

Objective-C底層探究之block(一)

從前面我們知道了block調(diào)用其實(shí)就是函數(shù)的調(diào)用。block本身用結(jié)構(gòu)體做了一些封裝聘裁。那現(xiàn)在又有一個(gè)疑問。block中可以無縫使用外部的變量蔚晨。加上__block關(guān)鍵字還可以在內(nèi)部修改變量又是怎么實(shí)現(xiàn)呢
我們可以把前面的代碼改一下:

int main(int argc, const char * argv[]) {
    @autoreleasepool {
        int a = 10;
        int b = 11;
        int c = 13;
        int (^block)(int,int) = ^(int a, int b) {
            return a+b+c;
        };
        block(a,b);
    }
    return 0;
}

同樣使用命令解析得到C++代碼

struct __main_block_impl_0 {
  struct __block_impl impl;
  struct __main_block_desc_0* Desc;
  int c;
  __main_block_impl_0(void *fp, struct __main_block_desc_0 *desc, int _c, int flags=0) : c(_c) {
    impl.isa = &_NSConcreteStackBlock;
    impl.Flags = flags;
    impl.FuncPtr = fp;
    Desc = desc;
  }
};
static int __main_block_func_0(struct __main_block_impl_0 *__cself, int a, int b) {
  int c = __cself->c; // bound by copy

            return a+b+c;
        }

static struct __main_block_desc_0 {
  size_t reserved;
  size_t Block_size;
} __main_block_desc_0_DATA = { 0, sizeof(struct __main_block_impl_0)};
int main(int argc, const char * argv[]) {
    /* @autoreleasepool */ { __AtAutoreleasePool __autoreleasepool; 
        int a = 10;
        int b = 11;
        int c = 13;
        int (*block)(int,int) = ((int (*)(int, int))&__main_block_impl_0((void *)__main_block_func_0, &__main_block_desc_0_DATA, c));
        ((int (*)(__block_impl *, int, int))((__block_impl *)block)->FuncPtr)((__block_impl *)block, a, b);
    }
    return 0;
}

我們來看看與之前例子不同的部分:

struct __main_block_impl_0 {
  struct __block_impl impl;
  struct __main_block_desc_0* Desc;
  int c;
  __main_block_impl_0(void *fp, struct __main_block_desc_0 *desc, int _c, int flags=0) : c(_c) {
    impl.isa = &_NSConcreteStackBlock;
    impl.Flags = flags;
    impl.FuncPtr = fp;
    Desc = desc;
  }
};
static int __main_block_func_0(struct __main_block_impl_0 *__cself, int a, int b) {
  int c = __cself->c; // bound by copy

            return a+b+c;
        }

發(fā)現(xiàn)__main_block_impl_0結(jié)構(gòu)體里面多了一個(gè)變量c而在block創(chuàng)建的時(shí)候?qū)?strong>c變量通過構(gòu)造函數(shù)傳了進(jìn)來

int (*block)(int,int) = ((int (*)(int, int))&__main_block_impl_0((void *)__main_block_func_0, &__main_block_desc_0_DATA, c));

也就是說c變量實(shí)質(zhì)上是做了一層拷貝儿礼。而在函數(shù)調(diào)用時(shí)通過__cself參數(shù)拿到block的指針并把c拷貝一份局部變量后使用咖杂。
那加了__block關(guān)鍵字之后是怎么實(shí)現(xiàn)在內(nèi)部改變外部值的呢,我們修改代碼如下:

int main(int argc, const char * argv[]) {
    @autoreleasepool {
        int a = 10;
        int b = 11;
        __block int c = 13;
        int (^block)(int,int) = ^(int a, int b) {
            c = 14;
            return a+b+c;
        };
        block(a,b);
    }
    return 0;
}

同樣使用命令解析:

struct __Block_byref_c_0 {
  void *__isa;
__Block_byref_c_0 *__forwarding;
 int __flags;
 int __size;
 int c;
};

struct __main_block_impl_0 {
  struct __block_impl impl;
  struct __main_block_desc_0* Desc;
  __Block_byref_c_0 *c; // by ref
  __main_block_impl_0(void *fp, struct __main_block_desc_0 *desc, __Block_byref_c_0 *_c, int flags=0) : c(_c->__forwarding) {
    impl.isa = &_NSConcreteStackBlock;
    impl.Flags = flags;
    impl.FuncPtr = fp;
    Desc = desc;
  }
};
static int __main_block_func_0(struct __main_block_impl_0 *__cself, int a, int b) {
  __Block_byref_c_0 *c = __cself->c; // bound by ref

            (c->__forwarding->c) = 14;
            return a+b+(c->__forwarding->c);
        }
static void __main_block_copy_0(struct __main_block_impl_0*dst, struct __main_block_impl_0*src) {_Block_object_assign((void*)&dst->c, (void*)src->c, 8/*BLOCK_FIELD_IS_BYREF*/);}

static void __main_block_dispose_0(struct __main_block_impl_0*src) {_Block_object_dispose((void*)src->c, 8/*BLOCK_FIELD_IS_BYREF*/);}

static struct __main_block_desc_0 {
  size_t reserved;
  size_t Block_size;
  void (*copy)(struct __main_block_impl_0*, struct __main_block_impl_0*);
  void (*dispose)(struct __main_block_impl_0*);
} __main_block_desc_0_DATA = { 0, sizeof(struct __main_block_impl_0), __main_block_copy_0, __main_block_dispose_0};
int main(int argc, const char * argv[]) {
    /* @autoreleasepool */ { __AtAutoreleasePool __autoreleasepool; 
        int a = 10;
        int b = 11;
        __attribute__((__blocks__(byref))) __Block_byref_c_0 c = {(void*)0,(__Block_byref_c_0 *)&c, 0, sizeof(__Block_byref_c_0), 13};
        int (*block)(int,int) = ((int (*)(int, int))&__main_block_impl_0((void *)__main_block_func_0, &__main_block_desc_0_DATA, (__Block_byref_c_0 *)&c, 570425344));
        ((int (*)(__block_impl *, int, int))((__block_impl *)block)->FuncPtr)((__block_impl *)block, a, b);
    }
    return 0;
}

可以發(fā)現(xiàn)block里面多了一個(gè)指針 __Block_byref_c_0 類型的指針蚊夫。而block外面加了__block 標(biāo)記的變量c被解釋成 __Block_byref_c_0 類型的結(jié)構(gòu)體诉字。并在調(diào)用時(shí)將c的地址傳入block的構(gòu)造函數(shù)中。block解析的結(jié)構(gòu)體也多了一個(gè)指向__Block_byref_c_0對(duì)象的 c 指針用于儲(chǔ)存該地址知纷。這樣就可以在block結(jié)構(gòu)體內(nèi)改變c變量的值了壤圃。

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市琅轧,隨后出現(xiàn)的幾起案子伍绳,更是在濱河造成了極大的恐慌,老刑警劉巖乍桂,帶你破解...
    沈念sama閱讀 222,590評(píng)論 6 517
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件冲杀,死亡現(xiàn)場(chǎng)離奇詭異,居然都是意外死亡睹酌,警方通過查閱死者的電腦和手機(jī)权谁,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 95,157評(píng)論 3 399
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來憋沿,“玉大人旺芽,你說我怎么就攤上這事÷倍剩” “怎么了甥绿?”我有些...
    開封第一講書人閱讀 169,301評(píng)論 0 362
  • 文/不壞的土叔 我叫張陵字币,是天一觀的道長则披。 經(jīng)常有香客問我,道長洗出,這世上最難降的妖魔是什么士复? 我笑而不...
    開封第一講書人閱讀 60,078評(píng)論 1 300
  • 正文 為了忘掉前任,我火速辦了婚禮翩活,結(jié)果婚禮上阱洪,老公的妹妹穿的比我還像新娘。我一直安慰自己菠镇,他們只是感情好冗荸,可當(dāng)我...
    茶點(diǎn)故事閱讀 69,082評(píng)論 6 398
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著利耍,像睡著了一般蚌本。 火紅的嫁衣襯著肌膚如雪盔粹。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 52,682評(píng)論 1 312
  • 那天程癌,我揣著相機(jī)與錄音舷嗡,去河邊找鬼。 笑死嵌莉,一個(gè)胖子當(dāng)著我的面吹牛进萄,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播锐峭,決...
    沈念sama閱讀 41,155評(píng)論 3 422
  • 文/蒼蘭香墨 我猛地睜開眼中鼠,長吁一口氣:“原來是場(chǎng)噩夢(mèng)啊……” “哼!你這毒婦竟也來了沿癞?” 一聲冷哼從身側(cè)響起兜蠕,我...
    開封第一講書人閱讀 40,098評(píng)論 0 277
  • 序言:老撾萬榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎抛寝,沒想到半個(gè)月后熊杨,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 46,638評(píng)論 1 319
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡盗舰,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 38,701評(píng)論 3 342
  • 正文 我和宋清朗相戀三年晶府,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片钻趋。...
    茶點(diǎn)故事閱讀 40,852評(píng)論 1 353
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡川陆,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出蛮位,到底是詐尸還是另有隱情较沪,我是刑警寧澤,帶...
    沈念sama閱讀 36,520評(píng)論 5 351
  • 正文 年R本政府宣布失仁,位于F島的核電站尸曼,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏萄焦。R本人自食惡果不足惜控轿,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 42,181評(píng)論 3 335
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望拂封。 院中可真熱鬧茬射,春花似錦、人聲如沸冒签。這莊子的主人今日做“春日...
    開封第一講書人閱讀 32,674評(píng)論 0 25
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽萧恕。三九已至刚梭,卻和暖如春档悠,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背望浩。 一陣腳步聲響...
    開封第一講書人閱讀 33,788評(píng)論 1 274
  • 我被黑心中介騙來泰國打工辖所, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人磨德。 一個(gè)月前我還...
    沈念sama閱讀 49,279評(píng)論 3 379
  • 正文 我出身青樓缘回,卻偏偏與公主長得像,于是被迫代替她去往敵國和親典挑。 傳聞我的和親對(duì)象是個(gè)殘疾皇子酥宴,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 45,851評(píng)論 2 361

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