• <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>

            羅朝輝(飄飄白云)

            關(guān)注嵌入式操作系統(tǒng),移動平臺,圖形開發(fā)。-->加微博 ^_^

              C++博客 :: 首頁 :: 新隨筆 :: 聯(lián)系 :: 聚合  :: 管理 ::
              85 隨筆 :: 0 文章 :: 169 評論 :: 0 Trackbacks
            深入淺出Cocoa之類與對象
            羅朝輝(http://m.shnenglu.com/kesalin/
            CC許可,轉(zhuǎn)載請注明出處

            最近打算寫一些ObjC中比較底層的東西,尤其是 runtime 相關(guān)的。蘋果已經(jīng)將 ObjC runtime 代碼開源了,我們可以從:http://opensource.apple.com/source/objc4/objc4-493.9/runtime/ 瀏覽源代碼,或點此下載源代碼。

            從哪里入手呢?那當(dāng)然是最基本的類與對象。與C++相比,ObjC中的類與對象結(jié)構(gòu)要簡潔與一致得多(參考《深度探索C++對象模型》,你就知道C++中類與對象結(jié)構(gòu)的復(fù)雜)。本文將詳細(xì)講解ObjC中類與對象的結(jié)構(gòu),下回將講如何在 runtime 時操作類。

            我們可以在/usr/include/objc/objc.h 和 runtime.h  中找到對 class 與 object 的定義:

            typedef struct objc_class *Class;
            typedef struct objc_object {
                Class isa;
            } *id;

            Class 是一個 objc_class 結(jié)構(gòu)類型的指針;而 id(任意對象) 是一個 objc_object 結(jié)構(gòu)類型的指針,其第一個成員是一個 objc_class 結(jié)構(gòu)類型的指針。注意這里有一關(guān)鍵的引申解讀:內(nèi)存布局以一個 objc_class 指針為開始的所有東東都可以當(dāng)做一個 object 來對待! 那 objc_class 又是怎樣一個結(jié)構(gòu)體呢?且看:

            struct objc_class
            {
                struct objc_class* isa;
                struct objc_class* super_class;
                const char* name;
                long version;
                long info;
                long instance_size;
                struct objc_ivar_list* ivars;
                struct objc_method_list** methodLists;
                struct objc_cache* cache;
                struct objc_protocol_list* protocols;
            };

            objc_class 結(jié)構(gòu)體的各成員介紹如下:

            isa:是一個 objc_class 類型的指針,看到這里,想起我前面的引申解讀了沒?內(nèi)存布局以一個 objc_class 指針為開始的所有東東都可以當(dāng)做一個 object 來對待! 這就是說 objc_class 或者說類其實也可以當(dāng)做一個 objc_object 對象來對待!對象是對象,類也是對象,是不是有點混淆?別急,ObjC發(fā)明(or 重用)了一個術(shù)語來區(qū)分這兩種不同的對象:類對象(class object)與實例對象(instance object)。OK,名稱混淆的問題解決,下面我將使用這兩個術(shù)語來區(qū)分不同的對象,而使用“對象”這一術(shù)語來泛指所有的對象。ObjC還對類對象與實例對象中的 isa 所指向的類結(jié)構(gòu)作了不同的命名:類對象中的 isa 指向類結(jié)構(gòu)被稱作 metaclass,metaclass 存儲類的static類成員變量與static類成員方法(+開頭的方法);實例對象中的 isa 指向類結(jié)構(gòu)稱作 class(普通的),class 結(jié)構(gòu)存儲類的普通成員變量與普通成員方法(-開頭的方法)。

            super_class:一看就明白,指向該類的父類唄!如果該類已經(jīng)是最頂層的根類(如 NSObject 或 NSProxy),那么 super_class 就為 NULL。

            好,先中斷一下其他類結(jié)構(gòu)成員的介紹,讓我們厘清一下在繼承層次中,子類,父類,根類(這些都是普通 class)以及其對應(yīng)的 metaclass 的 isa 與 super_class 之間關(guān)系:
            規(guī)則一:類的實例對象的 isa 指向該類;該類的 isa 指向該類的 metaclass;
            規(guī)則二:類的 super_class 指向其父類,如果該類為根類則值為 NULL;
            規(guī)則三:metaclass 的 isa 指向根 metaclass,如果該 metaclass 是根 metaclass 則指向自身;
            規(guī)則四:metaclass 的 super_class 指向父 metaclass,如果該 metaclass 是根 metaclass 則指向該 metaclass 對應(yīng)的類;

            好吧,文字總是那么乏力,有圖有真相!

            <instance object,class,metaclass 的 isa 與 super_class 關(guān)系圖>

            那么 class 與 metaclass 有什么區(qū)別呢?

            class 是 instance object 的類類型。當(dāng)我們向?qū)嵗龑ο蟀l(fā)送消息(實例方法)時,我們在該實例對象的 class 結(jié)構(gòu)的 methodlists 中去查找響應(yīng)的函數(shù),如果沒找到匹配的響應(yīng)函數(shù)則在該 class 的父類中的 methodlists 去查找(查找鏈為上圖的中間那一排)。如下面的代碼中,向str 實例對象發(fā)送 lowercaseString 消息,會在 NSString 類結(jié)構(gòu)的 methodlists 中去查找 lowercaseString 的響應(yīng)函數(shù)。

            NSString * str;
            [str lowercaseString];

            metaclass 是 class object 的類類型。當(dāng)我們向類對象發(fā)送消息(類方法)時,我們在該類對象的 metaclass 結(jié)構(gòu)的 methodlists 中去查找響應(yīng)的函數(shù),如果沒有找到匹配的響應(yīng)函數(shù)則在該 metaclass 的父類中的 methodlists 去查找(查找鏈為上圖的最右邊那一排)。如下面的代碼中,向 NSString 類對象發(fā)送 stringWithString 消息,會在 NSString 的 metaclass 類結(jié)構(gòu)的 methodlists 中去查找 stringWithString 的響應(yīng)函數(shù)。

            [NSString stringWithString:@"str"];

            好,至此我們明白了類的結(jié)構(gòu)層次,讓我們接著看類結(jié)構(gòu)中的其他成員。

            name:一個 C 字符串,指示類的名稱。我們可以在運行期,通過這個名稱查找到該類(通過:id objc_getClass(const char *aClassName))或該類的 metaclass(id objc_getMetaClass(const char *aClassName));

            version:類的版本信息,默認(rèn)初始化為 0。我們可以在運行期對其進(jìn)行修改(class_setVersion)或獲?。╟lass_getVersion)。

            info:供運行期使用的一些位標(biāo)識。有如下一些位掩碼:
            CLS_CLASS (0x1L) 表示該類為普通 class ,其中包含實例方法和變量;
            CLS_META (0x2L) 表示該類為 metaclass,其中包含類方法;
            CLS_INITIALIZED (0x4L) 表示該類已經(jīng)被運行期初始化了,這個標(biāo)識位只被 objc_addClass 所設(shè)置;
            CLS_POSING (0x8L) 表示該類被 pose 成其他的類;(poseclass 在ObjC 2.0中被廢棄了);
            CLS_MAPPED (0x10L) 為ObjC運行期所使用
            CLS_FLUSH_CACHE (0x20L) 為ObjC運行期所使用
            CLS_GROW_CACHE (0x40L) 為ObjC運行期所使用
            CLS_NEED_BIND (0x80L) 為ObjC運行期所使用
            CLS_METHOD_ARRAY (0x100L) 該標(biāo)志位指示 methodlists 是指向一個 objc_method_list 還是一個包含 objc_method_list 指針的數(shù)組;

            instance_size:該類的實例變量大小(包括從父類繼承下來的實例變量);


            ivars:指向 objc_ivar_list 的指針,存儲每個實例變量的內(nèi)存地址,如果該類沒有任何實例變量則為 NULL;

            methodLists:與 info 的一些標(biāo)志位有關(guān),CLS_METHOD_ARRAY 標(biāo)識位決定其指向的東西(是指向單個 objc_method_list還是一個 objc_method_list 指針數(shù)組),如果 info 設(shè)置了 CLS_CLASS 則 objc_method_list  存儲實例方法,如果設(shè)置的是 CLS_META 則存儲類方法;

            cache:指向 objc_cache 的指針,用來緩存最近使用的方法,以提高效率;

            protocols:指向 objc_protocol_list 的指針,存儲該類聲明要遵守的正式協(xié)議。

            總結(jié)
            ObjC 為每個類的定義生成兩個 objc_class ,一個即普通的 class,另一個即 metaclass。我們可以在運行期創(chuàng)建這兩個 objc_class 數(shù)據(jù)結(jié)構(gòu),然后使用 objc_addClass 動態(tài)地創(chuàng)建新的類定義。這個夠動態(tài)夠強大的吧?下回講演示如何在運行期動態(tài)創(chuàng)建新類。
            posted on 2012-01-19 16:58 羅朝輝 閱讀(4711) 評論(5)  編輯 收藏 引用 所屬分類: Cocoa 開發(fā)

            評論

            # re: [Cocoa]深入淺出Cocoa之類與對象 2012-01-20 19:40 megax
            寫的不錯!  回復(fù)  更多評論
              

            # re: [Cocoa]深入淺出Cocoa之類與對象 2012-01-22 08:06 marvin
            http://www.cnblogs.com/studentdeng/archive/2011/10/06/2199873.html

            有興趣可以看看此人的博,和你的差不多  回復(fù)  更多評論
              

            # re: [Cocoa]深入淺出Cocoa之類與對象 2012-01-30 12:40 羅朝輝
            @marvin
            謝謝~~  回復(fù)  更多評論
              

            # re: [Cocoa]深入淺出Cocoa之類與對象 2013-01-22 09:49 舍得333
            講的還比較透徹,支持  回復(fù)  更多評論
              

            # re: [Cocoa]深入淺出Cocoa之類與對象[未登錄] 2016-04-08 11:33 sam
            對于類對象的iVars有個疑問:
            按我的理解的話,實例對象里面應(yīng)該存儲的是個體的實例變量值,類對象中的iVars應(yīng)該只是實例變量的描述,相當(dāng)于類型定義吧。

            所以不知道樓主在文中所說的“class 結(jié)構(gòu)存儲類的普通成員變量”這個怎么解?  回復(fù)  更多評論
              

            久久久久亚洲AV成人网人人网站 | 久久影院午夜理论片无码 | 99精品国产在热久久无毒不卡 | 国产精品99久久99久久久| 久久国产高潮流白浆免费观看| 大伊人青草狠狠久久| 狠狠色丁香婷婷综合久久来来去 | 久久精品无码免费不卡| 国产精品久久精品| 久久精品国产黑森林| 久久99久久99精品免视看动漫| 久久成人影院精品777| 亚洲一区精品伊人久久伊人| 国产成人精品久久| 久久久久亚洲AV无码永不| 久久久久亚洲AV无码专区网站| 日韩av无码久久精品免费| 99精品国产免费久久久久久下载| 亚洲AV无码1区2区久久| 亚洲精品成人网久久久久久| 久久久久99精品成人片试看| 国产精品久久久久久久app| 99久久婷婷国产一区二区| 青青草国产成人久久91网| 亚洲国产精品婷婷久久| 亚洲国产欧美国产综合久久| 人妻精品久久久久中文字幕| 2020久久精品亚洲热综合一本| 亚洲精品国产成人99久久| 青青草国产精品久久| 99国产欧美久久久精品蜜芽| 亚洲国产精品久久电影欧美| 亚洲va久久久久| 亚洲精品乱码久久久久66| 久久久久久久精品妇女99| 精品少妇人妻av无码久久| 伊人久久大香线蕉亚洲五月天| 色欲综合久久躁天天躁蜜桃| 热99RE久久精品这里都是精品免费| 国产真实乱对白精彩久久| 久久亚洲AV无码西西人体|