• <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 之動態(tài)創(chuàng)建類
            CC許可,轉(zhuǎn)載請注明出處

            在前文《深入淺出Cocoa之類與對象》一文中,我已經(jīng)詳細(xì)介紹了ObjC中的 Class 與 Object 的概念,今天我們來如何在運(yùn)行
            時(shí)
            動態(tài)創(chuàng)建類。下面這個(gè)函數(shù)就是應(yīng)用前面講到的Class,MetaClass的概念,在運(yùn)行時(shí)動態(tài)創(chuàng)建一個(gè)類。這個(gè)函數(shù)來自《Inside Mac OS X-The Objective-C Programming Language》。

            #import <objc/objc.h>
            #import <objc/runtime.h>

            BOOL CreateClassDefinition( const char * name, const char * superclassName)
            {
                struct objc_class * meta_class;
                struct objc_class * super_class;
                struct objc_class * new_class;
                struct objc_class * root_class;
                va_list args;
                
                // 確保父類存在
                super_class = (struct objc_class *)objc_lookUpClass (superclassName);
                if (super_class == nil)
                {
                    return NO;
                }
                
                // 確保要創(chuàng)建的類不存在
                if (objc_lookUpClass (name) != nil)
                {
                    return NO;
                }
                
                // 查找 root class,因?yàn)?meta class 的 isa 指向 root class 的 meta class
                root_class = super_class;
                while( root_class->super_class != nil )
                {
                    root_class = root_class->super_class;
                }
                
                // 為 class 及其 meta class 分配內(nèi)存
                new_class = calloc( 2, sizeof(struct objc_class) );
                meta_class = &new_class[1];
                
                // 設(shè)置 class
                new_class->isa = meta_class;
                new_class->info = CLS_CLASS;
                meta_class->info = CLS_META;

                
            // 拷貝類名字,這里為了提高效率,讓 class 與 meta class 都指向同一個(gè)類名字符串
                new_class->name = malloc (strlen (name) + 1);
                strcpy ((char*)new_class->name, name);
                meta_class->name = new_class->name;
                
                
            // 分配并置空 method lists,我們可以在之后使用 
            class_addMethods 向類中增加方法

                new_class->methodLists = calloc( 1, sizeof(struct objc_method_list *) );
                meta_class->methodLists = calloc( 1, sizeof(struct objc_method_list *) );
                
                
            // 將類加入到繼承體系中去:
                
            // 1,設(shè)置類的 super class
                
            // 2,設(shè)置 meta class 的 super class
                
            // 3,設(shè)置 meta class 的 isa
                new_class->super_class = super_class;
                meta_class->super_class = super_class->isa;
                meta_class->isa = (void *)root_class->isa;
                
                // 最后,將 class 注冊到運(yùn)行時(shí)系統(tǒng)中
                objc_addClass( new_class );
                
                return YES;
            }

            如果要在代碼中使用運(yùn)行時(shí)相關(guān)的函數(shù),我們需要導(dǎo)入 libobjc.dylib,并導(dǎo)入相關(guān)的頭文件(比如這里的 runtime.h)。

            前文中總結(jié)到“ObjC 為每個(gè)類的定義生成兩個(gè) objc_class ,一個(gè)即普通的 class,另一個(gè)即 metaclass。我們可以在運(yùn)行期創(chuàng)建這兩個(gè) objc_class 數(shù)據(jù)結(jié)構(gòu),然后使用 objc_addClass 動態(tài)地創(chuàng)建新的類定義。”,這在上面的代碼中就體現(xiàn)出來了:new_class 和 meta_class 就是新類所必須的兩個(gè) objc_class。其他的代碼就不解釋了,注釋以及代碼足以自明了。

            在實(shí)際的運(yùn)用中,我們使用 ObjC 運(yùn)行時(shí)函數(shù)來動態(tài)創(chuàng)建類:
            Class objc_allocateClassPair(Class superclass, const char *name, size_t extraBytes);

            譬如:
            #import <objc/objc.h>
            #import <objc/runtime.h>

            void ReportFunction(id self, SEL _cmd)
            {
                NSLog(@" >> This object is %p.", self);
                NSLog(@" >> Class is %@, and super is %@.", [self class], [self superclass]);
                
                Class prevClass = NULL;
                int count = 1;
                for (Class currentClass = [self class]; currentClass; ++count)
                {
                    prevClass = currentClass;
                    
                    NSLog(@" >> Following the isa pointer %d times gives %p", count, currentClass);
                    
                    currentClass = object_getClass(currentClass);
                    if (prevClass == currentClass)
                        break;
                }
                
                NSLog(@" >> NSObject's class is %p", [NSObject class]);
                NSLog(@" >> NSObject's meta class is %p", object_getClass([NSObject class]));
            }

            int main (int argc, const char * argv[])
            {

                @autoreleasepool
                {
                    Class newClass = objc_allocateClassPair([NSString class], "NSStringSubclass", 0);
                    class_addMethod(newClass, @selector(report), (IMP)ReportFunction, "v@:");
                    objc_registerClassPair(newClass);
                    
                    id instanceOfNewClass = [[newClass alloc] init];
                    [instanceOfNewClass performSelector:@selector(report)];
                    [instanceOfNewClass release];
                }
                
                return 0;
            }

            在上面的代碼中,我們創(chuàng)建繼承自 NSString 的子類 NSStringSubclass,然后向其中添加方法 report,并在運(yùn)行時(shí)系統(tǒng)中注冊,這樣我們就可以使用這個(gè)新類了。在這里使用 performSelector 來向新類的對象發(fā)送消息,可以避免編譯警告信息(因?yàn)槲覀儾]有聲明該類及其可響應(yīng)的消息)。

            執(zhí)行結(jié)果為:
             >> This object is 0x100114710.
             >> Class is NSStringSubclass, and super is NSString.
             >> Following the isa pointer 1 times gives 0x100114410
             >> Following the isa pointer 2 times gives 0x100114560
             >> Following the isa pointer 3 times gives 0x7fff7e257b50
             >> NSObject's class is 0x7fff7e257b78
             >> NSObject's meta class is 0x7fff7e257b50

            根據(jù)前文中的類關(guān)系圖,我們不難從執(zhí)行結(jié)果中分析出 NSStringSubclass 的內(nèi)部類結(jié)構(gòu):
            1,對象的地址為 :0x100114710
            2,class 的地址為:0x100114410
            3,meta class 的地址為:0x100114560
            4,meta class 的 class 地址為:0x7fff7e257b50 (也是 NSObject 的 meta class)
            5,NSObject 的 meta class 的 meta class 是其自身
            posted on 2012-01-30 14:05 羅朝輝 閱讀(4649) 評論(2)  編輯 收藏 引用 所屬分類: 移動開發(fā)Cocoa 開發(fā)

            評論

            # re: [Cocoa]深入淺出Cocoa 之動態(tài)創(chuàng)建類 2012-01-30 15:45 marvin
            不錯(cuò)
            objc的這種動態(tài)性,類似于動態(tài)語言,其它編譯型語言都很難做到  回復(fù)  更多評論
              

            # re: [Cocoa]深入淺出Cocoa 之動態(tài)創(chuàng)建類 2012-08-11 10:04 xdream
            @marvin
            擺脫,oc本來就是動態(tài)語言  回復(fù)  更多評論
              

            无码人妻久久一区二区三区蜜桃| 久久久无码精品亚洲日韩软件| 久久精品国产第一区二区| 韩国免费A级毛片久久| 国产午夜福利精品久久2021| 热久久最新网站获取| 精品无码久久久久久午夜| 99久久国产宗和精品1上映| 久久久久久久人妻无码中文字幕爆| 亚洲精品乱码久久久久66| 久久精品国产久精国产| 欧美日韩精品久久久久| 久久精品一区二区影院 | 国产69精品久久久久99| 久久九九免费高清视频| 观看 国产综合久久久久鬼色 欧美 亚洲 一区二区 | 狠狠色丁香久久婷婷综合_中 | 国产精品中文久久久久久久| 日批日出水久久亚洲精品tv| 久久免费国产精品| 精品久久久一二三区| 亚洲国产精品成人久久| AV无码久久久久不卡蜜桃| 久久这里只精品国产99热| 美女久久久久久| 久久婷婷色香五月综合激情| 久久久久se色偷偷亚洲精品av| 久久精品一区二区三区不卡| 久久最新精品国产| 性欧美大战久久久久久久| 久久久久久午夜成人影院| 9久久9久久精品| 精品国产乱码久久久久久郑州公司 | 久久夜色精品国产亚洲av| 久久精品极品盛宴观看| 国产成人精品久久二区二区| 久久婷婷五月综合色99啪ak| 久久精品人人做人人妻人人玩| 国产精品成人精品久久久| 中文字幕人妻色偷偷久久| 国产精品永久久久久久久久久|