主頁 > 移動端開發 > Objective-C block 底層詳解

Objective-C block 底層詳解

2021-09-15 10:09:51 移動端開發

筆者前幾天做了一道筆試題:默認情況下 block 是不能修改外面的 auto 變數的,解決辦法? 思索了好久才想起可以用 __block 或者 static 修飾符來修飾 auto 變數,看來有必要好好探索一下 block 的相關知識,

目錄

1. block 的使用

1.1 什么是 block ?

block 的宣告

block 的定義

block 的呼叫

2. block 的底層資料結構

3. block 的變數捕獲機制

3.1 auto 自動變數

3.2 static 型別的區域變數

3.3 全域變數

為什么區域變數需要捕獲,而全域變數不用呢?

3.4 __block 修飾的變數

3.4.1 __block 的使用:

3.4.2 __block 修飾符


???????

1. block 的使用


1.1 什么是 block ?

塊,封裝了函式呼叫以及呼叫環境的 OC 物件,

  • block 的宣告

// 1.
@property (nonatomic, copy) void(^myBlock1)(void);
// 2.BlockType:型別別名
typedef void(^BlockType)(void);
@property (nonatomic, copy) BlockType myBlock2;
// 3.
    // 回傳值型別(^block變數名)(引數1型別,引數2型別,...)
    void(^block)(void);
  • block 的定義

    // ^回傳值型別(引數1,引數2,...){};
    // 1.無回傳值,無引數
    void(^block1)(void) = ^{
        
    };
    // 2.無回傳值,有引數
    void(^block2)(int) = ^(int a){
        
    };
    // 3.有回傳值,無引數(不管有沒有回傳值,定義的回傳值型別都可以省略)
    int(^block3)(void) = ^int{
        return 3;
    };
    // 以上Block的定義也可以這樣寫:
    int(^block4)(void) = ^{
        return 3;
    };
    // 4.有回傳值,有引數
    int(^block5)(int) = ^int(int a){
        return 3 * a;
    };
  • block 的呼叫

    // 1.無回傳值,無引數
    block1();
    // 2.有回傳值,有引數
    int a = block5(2);

2. block 的底層資料結構


通過 Clang 將以下的 OC 代碼轉換為 C++代碼

// Clang
xcrun -sdk iphoneos clang -arch arm64 -rewrite-objc main.m
//main.m
#import <Foundation/Foundation.h>

int main(int argc, const char * argv[]) {
    @autoreleasepool {
        void(^block)(void) = ^{
            NSLog(@"我是 block");
        };
        block();
    }
    return 0;
}

轉換為 C++ 代碼 :

struct __main_block_impl_0 {
  struct __block_impl impl;// block 結構體
  struct __main_block_desc_0* Desc;// block 的描述物件
/*
block 的建構式
 ** 回傳值:__main_block_impl_0 結構體
 ** 引數一:__main_block_func_0 結構體
 ** 引數二:__main_block_desc_0 結構體的地址
 ** 引數三:flags 標識位
*/
  __main_block_impl_0(void *fp, struct __main_block_desc_0 *desc, int flags=0) {
    impl.isa = &_NSConcreteStackBlock;// &_NSConcreteStackBlock 表示存盤在堆疊上
    impl.Flags = flags;
    impl.FuncPtr = fp;
    Desc = desc;
  }
};

//block 結構體
struct __block_impl {
  void *isa;//block 的型別
  int Flags;
  int Reserved;
  void *FuncPtr;// block的執行函式指標,指向__main_block_func_0
};

//封裝了 block 中的代碼
static void __main_block_func_0(struct __main_block_impl_0 *__cself) {

            NSLog((NSString *)&__NSConstantStringImpl__var_folders_yx_7jg_wdg128v45l4cn_1g265h0000gn_T_main_03dcda_mi_0);
        }


static struct __main_block_desc_0 {
  size_t reserved;
  size_t Block_size;// block 所占的記憶體空間
} __main_block_desc_0_DATA = { 0, sizeof(struct __main_block_impl_0)};
int main(int argc, const char * argv[]) {
    /* @autoreleasepool */ { __AtAutoreleasePool __autoreleasepool;
/*
         ** void(^block)(void) = ^{
                NSLog(@"呼叫了block");
            };
         ** 定義block的本質:
         ** 呼叫__main_block_impl_0()建構式
         ** 并且給它傳了兩個引數 __main_block_func_0 和 &__main_block_desc_0_DATA
         ** __main_block_func_0 封裝了block里的代碼
         ** 拿到函式的回傳值,再取回傳值的地址 &__main_block_impl_0,
         ** 把這個地址賦值給 block
         */
        void(*block)(void) = ((void (*)())&__main_block_impl_0((void *)__main_block_func_0, &__main_block_desc_0_DATA));
/*
         ** block();
         ** 呼叫block的本質:
         ** 通過 __main_block_impl_0 中的 __block_impl 中的 FuncPtr 拿到函式地址,直接呼叫
         */    
        ((void (*)(__block_impl *))((__block_impl *)block)->FuncPtr)((__block_impl *)block);
    }
    return 0;
}

3. block 的變數捕獲機制

  • 對于全域變數,不會捕獲到 block 內部,訪問方式為直接訪問;
  • 對于 auto 型別的區域變數,會捕獲到 block 內部,block 內部會自動生成一個成員變數,訪問方式為值傳遞;
  • 對于 static 修飾的區域變數,會捕獲到 block 內部,block 內部會自動生成一個成員變數,訪問方式為指標傳遞,
  • 對于物件型別的區域變數,block 會連同其所有權修飾符一起捕獲,

3.1 auto 自動變數

將以下 OC 代碼轉換為 C++ 代碼,并貼出部分變動代碼

int main(int argc, const char * argv[]) {
    @autoreleasepool {
        int age = 10;
        void(^block)(void) = ^{
            NSLog(@"%d",age);
        };
        block();
    }
    return 0;
}
struct __main_block_impl_0 {
  struct __block_impl impl;
  struct __main_block_desc_0* Desc;
  int age;
  __main_block_impl_0(void *fp, struct __main_block_desc_0 *desc, int _age, int flags=0) : age(_age) {
    impl.isa = &_NSConcreteStackBlock;
    impl.Flags = flags;
    impl.FuncPtr = fp;
    Desc = desc;
  }
};
static void __main_block_func_0(struct __main_block_impl_0 *__cself) {
  int age = __cself->age; // bound by copy

            NSLog((NSString *)&__NSConstantStringImpl__var_folders_yx_7jg_wdg128v45l4cn_1g265h0000gn_T_main_40c716_mi_0,age);
        }

可以看出:

  • 在 __main_block_impl_0 結構體中會自動生成一個相同的 age 變數
  • 建構式 __main_block_impl_0( ) 中多出了一個 age 引數,用來捕獲外部的變數

由于是傳遞方式為值傳遞,所以我們在 block 外部修飾 age 變數時,不會影響到 block 中的 age 變數

3.2 static 型別的區域變數

將以下 OC 代碼轉換為 C++ 代碼,并貼出部分變動代碼

    static int age = 10;
    void(^block)(void) = ^{
        NSLog(@"%d",age);
    };
    age = 20;
    block();
    // 20
struct __main_block_impl_0 {
  struct __block_impl impl;
  struct __main_block_desc_0* Desc;
  int *age;
  __main_block_impl_0(void *fp, struct __main_block_desc_0 *desc, int *_age, int flags=0) : age(_age) {
    impl.isa = &_NSConcreteStackBlock;
    impl.Flags = flags;
    impl.FuncPtr = fp;
    Desc = desc;
  }
};
static void __main_block_func_0(struct __main_block_impl_0 *__cself) {
  int *age = __cself->age; // bound by copy

            NSLog((NSString *)&__NSConstantStringImpl__var_folders_yx_7jg_wdg128v45l4cn_1g265h0000gn_T_main_cb7943_mi_0,(*age));
        }

可以看出:

  • __main_block_impl_0 結構體中生成了一個相同型別的 age 指標
  • __main_block_impl_0 建構式多了個引數,用來捕獲外部的 age 變數的地址

由于傳遞方式是指標傳遞,所以修改 區域變數 age 時,age 的值會隨之變化

3.3 全域變數

將以下 OC 代碼轉換為 C++ 代碼,并貼出部分變動代碼

int height = 10;
static int age = 20;
int main(int argc, const char * argv[]) {
    @autoreleasepool {
        void(^block)(void) = ^{
            NSLog(@"%d,%d",height,age);
        };
        block();
    }
    return 0;
}
int height = 10;
static int age = 20;

struct __main_block_impl_0 {
  struct __block_impl impl;
  struct __main_block_desc_0* Desc;
  __main_block_impl_0(void *fp, struct __main_block_desc_0 *desc, int flags=0) {
    impl.isa = &_NSConcreteStackBlock;
    impl.Flags = flags;
    impl.FuncPtr = fp;
    Desc = desc;
  }
};
static void __main_block_func_0(struct __main_block_impl_0 *__cself) {

            NSLog((NSString *)&__NSConstantStringImpl__var_folders_yx_7jg_wdg128v45l4cn_1g265h0000gn_T_main_7a340f_mi_0,height,age);
        }

可以看出:

  • __main_block_impl_0 結構體中,并沒有自動生成 age 或 height 全域變數,也就是說沒有將變數捕獲到 block 內部

那么就有一個問題:

為什么區域變數需要捕獲,而全域變數不用呢?

  • 作用域的原因,全域變數哪里都可以直接訪問,所以不用捕獲;
  • 區域變數,外部不能直接訪問,所以需要捕獲;
  • auto 型別的區域變數可能會銷毀,其記憶體會消失,block 將來執行代碼的時候不可能再去訪問那塊記憶體,所以捕獲其值;
  • static 變數會一直保存在記憶體中, 所以捕獲其地址即可,

3.4 __block 修飾的變數


3.4.1 __block 的使用:

終于來到那個問題:

默認情況下 block 是不能修改外面的 auto 變數的,解決辦法?

  • 變數用 static 修飾(原因:捕獲 static 型別的區域變數是指標傳遞,可以訪問到該變數的記憶體地址)
  • 全域變數
  • __block(我們只希望臨時用一下這個變數臨時改一下而已,而改為 static 變數和全域變數會一直在記憶體中)

3.4.2 __block 修飾符

  • __block 可以用于解決 block 內部無法修改 auto 變數值的問題;
  • __block 不能修飾全域變數、靜態變數;
  • 編譯器會將 __block 變數包裝成一個物件(struct __Block_byref_age_0(byref:按地址傳遞));
  • 加 __block 修飾不會修改變數的性質,它還是 auto 變數;
  • 一般情況下,對被捕獲變數進行賦值(賦值!=使用)操作需要添加 __block 修飾符,比如給陣列添加或者洗掉物件,就不用加 __block 修飾;
  • 在 MRC 下使用 __block 修飾物件型別,在 block 內部不會對該物件進行 retain 操作,所以在 MRC 環境下可以通過 __block 解決回圈參考的問題,

將以下 OC 代碼轉換為 C++ 代碼,并貼出部分變動代碼

int main(int argc, const char * argv[]) {
    @autoreleasepool {
        __block int age = 10;
        void(^block)(void) = ^{
            age = 20;
            NSLog(@"block-%d",age);
        };
        block();
        NSLog(@"%d",age);
    }
    return 0;
}
struct __Block_byref_age_0 {
  void *__isa;
__Block_byref_age_0 *__forwarding;
 int __flags;
 int __size;
 int age;
};

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

            (age->__forwarding->age) = 20;
            NSLog((NSString *)&__NSConstantStringImpl__var_folders_yx_7jg_wdg128v45l4cn_1g265h0000gn_T_main_75529b_mi_0,(age->__forwarding->age));
        }
static void __main_block_copy_0(struct __main_block_impl_0*dst, struct __main_block_impl_0*src) {_Block_object_assign((void*)&dst->age, (void*)src->age, 8/*BLOCK_FIELD_IS_BYREF*/);}

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

可以看出:

  • 編譯器會將 __block 修飾的變數包裝成一個__Block_byref_age_0物件;
  • 以上age = 20;的賦值程序為:通過 block 結構體里的(__Block_byref_age_0)型別的 age 指標,找到__Block_byref_age_0結構體的記憶體(即被 __block 包裝成物件的記憶體),把__Block_byref_age_0結構體里的 age 變數的值改為20,

轉載請註明出處,本文鏈接:https://www.uj5u.com/yidong/300277.html

標籤:其他

上一篇:【Android 教程系列第 9 篇】升級 Android Studio 到 Arctic Fox 2020.3.1 后,使用搜索功能時出現中文亂碼的問題

下一篇:面試必備攻略!爆火的《超全Android App性能優化 &網路優化知識技能手冊》!

標籤雲
其他(157675) Python(38076) JavaScript(25376) Java(17977) C(15215) 區塊鏈(8255) C#(7972) AI(7469) 爪哇(7425) MySQL(7132) html(6777) 基礎類(6313) sql(6102) 熊猫(6058) PHP(5869) 数组(5741) R(5409) Linux(5327) 反应(5209) 腳本語言(PerlPython)(5129) 非技術區(4971) Android(4554) 数据框(4311) css(4259) 节点.js(4032) C語言(3288) json(3245) 列表(3129) 扑(3119) C++語言(3117) 安卓(2998) 打字稿(2995) VBA(2789) Java相關(2746) 疑難問題(2699) 细绳(2522) 單片機工控(2479) iOS(2429) ASP.NET(2402) MongoDB(2323) 麻木的(2285) 正则表达式(2254) 字典(2211) 循环(2198) 迅速(2185) 擅长(2169) 镖(2155) 功能(1967) .NET技术(1958) Web開發(1951) python-3.x(1918) HtmlCss(1915) 弹簧靴(1913) C++(1909) xml(1889) PostgreSQL(1872) .NETCore(1853) 谷歌表格(1846) Unity3D(1843) for循环(1842)

熱門瀏覽
  • 【從零開始擼一個App】Dagger2

    Dagger2是一個IOC框架,一般用于Android平臺,第一次接觸的朋友,一定會被搞得暈頭轉向。它延續了Java平臺Spring框架代碼碎片化,注解滿天飛的傳統。嘗試將各處代碼片段串聯起來,理清思緒,真不是件容易的事。更不用說還有各版本細微的差別。 與Spring不同的是,Spring是通過反射 ......

    uj5u.com 2020-09-10 06:57:59 more
  • Flutter Weekly Issue 66

    新聞 Flutter 季度調研結果分享 教程 Flutter+FaaS一體化任務編排的思考與設計 詳解Dart中如何通過注解生成代碼 GitHub 用對了嗎?Flutter 團隊分享如何管理大型開源專案 插件 flutter-bubble-tab-indicator A Flutter librar ......

    uj5u.com 2020-09-10 06:58:52 more
  • Proguard 常用規則

    介紹 Proguard 入口,如何查看輸出,如何使用 keep 設定入口以及使用實體,如何配置壓縮,混淆,校驗等規則。

    ......

    uj5u.com 2020-09-10 06:59:00 more
  • Android 開發技術周報 Issue#292

    新聞 Android即將獲得類AirDrop功能:可向附近設備快速分享檔案 谷歌為安卓檔案管理應用引入可安全隱藏資料的Safe Folder功能 Android TV新主界面將顯示電影、電視節目和應用推薦內容 泄露的Android檔案暗示了傳說中的谷歌Pixel 5a與折疊屏新機 谷歌發布Andro ......

    uj5u.com 2020-09-10 07:00:37 more
  • AutoFitTextureView Error inflating class

    報錯: Binary XML file line #0: Binary XML file line #0: Error inflating class xxx.AutoFitTextureView 解決: <com.example.testy2.AutoFitTextureView android: ......

    uj5u.com 2020-09-10 07:00:41 more
  • 根據Uri,Cursor沒有獲取到對應的屬性

    Android: 背景:呼叫攝像頭,拍攝視頻,指定保存的地址,但是回傳的Cursor檔案,只有名稱和大小的屬性,沒有其他諸如時長,連ID屬性都沒有 使用 cursor.getInt(cursor.getColumnIndexOrThrow(MediaStore.Video.Media.DURATIO ......

    uj5u.com 2020-09-10 07:00:44 more
  • Android連載29-持久化技術

    一、持久化技術 我們平時所使用的APP產生的資料,在記憶體中都是瞬時的,會隨著斷電、關機等丟失資料,因此android系統采用了持久化技術,用于存盤這些“瞬時”資料 持久化技術包括:檔案存盤、SharedPreference存盤以及資料庫存盤,還有更復雜的SD卡記憶體儲。 二、檔案存盤 最基本存盤方式, ......

    uj5u.com 2020-09-10 07:00:47 more
  • Android Camera2Video整合到自己專案里

    背景: Android專案里呼叫攝像頭拍攝視頻,原本使用的 MediaStore.ACTION_VIDEO_CAPTURE, 后來因專案需要,改成了camera2 1.Camera2Video 官方demo有點問題,下載后,不能直接整合到專案 問題1.多次拍攝視頻崩潰 問題2.雙擊record按鈕, ......

    uj5u.com 2020-09-10 07:00:50 more
  • Android 開發技術周報 Issue#293

    新聞 谷歌為Android TV開發者提供多種新功能 Android 11將自動填表功能整合到鍵盤輸入建議中 谷歌宣布Android Auto即將支持更多的導航和數字停車應用 谷歌Pixel 5只有XL版本 搭載驍龍765G且將比Pixel 4更便宜 [圖]Wear OS將迎來重磅更新:應用啟動時間 ......

    uj5u.com 2020-09-10 07:01:38 more
  • 海豚星空掃碼投屏 Android 接收端 SDK 集成 六步驟

    掃碼投屏,開放網路,獨占設備,不需要額外下載軟體,微信掃碼,發現設備。支持標準DLNA協議,支持倍速播放。視頻,音頻,圖片投屏。好點意思。還支持自定義基于 DLNA 擴展的操作動作。好像要收費,沒體驗。 這里簡單記錄一下集成程序。 一 跟目錄的build.gradle添加私有mevan倉庫 mave ......

    uj5u.com 2020-09-10 07:01:43 more
最新发布
  • 歡迎頁輪播影片

    如圖,引導開始,球從上落下,同時淡入文字,然后文字開始輪播,最后一頁時停止,點擊進入首頁。 在來看看效果圖。 重力球先不講,主要歡迎輪播簡單實作 首先新建一個類 TextTranslationXGuideView,用于影片展示 文本是類似的,最后會有個圖片箭頭影片,布局很簡單,就是一個 TextVi ......

    uj5u.com 2023-04-20 08:40:31 more
  • 【FAQ】關于華為推送服務因營銷訊息頻次管控導致服務通訊類訊息

    一. 問題描述 使用華為推送服務下發IM訊息時,下發訊息請求成功且code碼為80000000,但是手機總是收不到訊息; 在華為推送自助分析(Beta)平臺查看發現,訊息發送觸發了頻控。 二. 問題原因及背景 2023年1月05日起,華為推送服務對咨詢營銷類訊息做了單個設備每日推送數量上限管理,具體 ......

    uj5u.com 2023-04-20 08:40:11 more
  • 歡迎頁輪播影片

    如圖,引導開始,球從上落下,同時淡入文字,然后文字開始輪播,最后一頁時停止,點擊進入首頁。 在來看看效果圖。 重力球先不講,主要歡迎輪播簡單實作 首先新建一個類 TextTranslationXGuideView,用于影片展示 文本是類似的,最后會有個圖片箭頭影片,布局很簡單,就是一個 TextVi ......

    uj5u.com 2023-04-20 08:39:36 more
  • 【FAQ】關于華為推送服務因營銷訊息頻次管控導致服務通訊類訊息

    一. 問題描述 使用華為推送服務下發IM訊息時,下發訊息請求成功且code碼為80000000,但是手機總是收不到訊息; 在華為推送自助分析(Beta)平臺查看發現,訊息發送觸發了頻控。 二. 問題原因及背景 2023年1月05日起,華為推送服務對咨詢營銷類訊息做了單個設備每日推送數量上限管理,具體 ......

    uj5u.com 2023-04-20 08:39:13 more
  • iOS從UI記憶體地址到讀取成員變數(oc/swift)

    開發除錯時,我們發現bug時常首先是從UI顯示發現例外,下一步才會去定位UI相關連的資料的。XCode有給我們提供一系列debug工具,但是很多人可能還沒有形成一套穩定的除錯流程,因此本文嘗試解決這個問題,順便提出一個暴論:UI顯示例外問題只需要兩個步驟就能完成定位作業的80%: 定位例外 UI 組 ......

    uj5u.com 2023-04-19 09:16:23 more
  • FIDE重磅更新!性能飛躍!體驗有禮!

    FIDE 開發者工具重構升級啦!實作500%性能提升,誠邀體驗! 一直以來不少開發者朋友在社區反饋,在使用 FIDE 工具的程序中,時常會遇到諸如加載不及時、代碼預覽/渲染性能不如意的情況,十分影響開發體驗。 作為技術團隊,我們深知一件趁手的開發工具對開發者的重要性,因此,在2023年開年,FinC ......

    uj5u.com 2023-04-19 09:16:15 more
  • 游戲內嵌社區服務開放,助力開發者提升玩家互動與留存

    華為 HMS Core 游戲內嵌社區服務提供快速訪問華為游戲中心論壇能力,支持玩家直接在游戲內瀏覽帖子和交流互動,助力開發者擴展內容生產和觸達的場景。 一、為什么要游戲內嵌社區? 二、游戲內嵌社區的典型使用場景 1、游戲內打開論壇 您可以在游戲內繪制論壇入口,為玩家提供沉浸式發帖、瀏覽、點贊、回帖、 ......

    uj5u.com 2023-04-19 09:15:46 more
  • iOS從UI記憶體地址到讀取成員變數(oc/swift)

    開發除錯時,我們發現bug時常首先是從UI顯示發現例外,下一步才會去定位UI相關連的資料的。XCode有給我們提供一系列debug工具,但是很多人可能還沒有形成一套穩定的除錯流程,因此本文嘗試解決這個問題,順便提出一個暴論:UI顯示例外問題只需要兩個步驟就能完成定位作業的80%: 定位例外 UI 組 ......

    uj5u.com 2023-04-19 09:14:53 more
  • FIDE重磅更新!性能飛躍!體驗有禮!

    FIDE 開發者工具重構升級啦!實作500%性能提升,誠邀體驗! 一直以來不少開發者朋友在社區反饋,在使用 FIDE 工具的程序中,時常會遇到諸如加載不及時、代碼預覽/渲染性能不如意的情況,十分影響開發體驗。 作為技術團隊,我們深知一件趁手的開發工具對開發者的重要性,因此,在2023年開年,FinC ......

    uj5u.com 2023-04-19 09:14:08 more
  • 游戲內嵌社區服務開放,助力開發者提升玩家互動與留存

    華為 HMS Core 游戲內嵌社區服務提供快速訪問華為游戲中心論壇能力,支持玩家直接在游戲內瀏覽帖子和交流互動,助力開發者擴展內容生產和觸達的場景。 一、為什么要游戲內嵌社區? 二、游戲內嵌社區的典型使用場景 1、游戲內打開論壇 您可以在游戲內繪制論壇入口,為玩家提供沉浸式發帖、瀏覽、點贊、回帖、 ......

    uj5u.com 2023-04-19 09:08:34 more