久久国产成人av_抖音国产毛片_a片网站免费观看_A片无码播放手机在线观看,色五月在线观看,亚洲精品m在线观看,女人自慰的免费网址,悠悠在线观看精品视频,一级日本片免费的,亚洲精品久,国产精品成人久久久久久久

分享

[Cocoa]深入淺出Cocoa 之動態(tài)創(chuàng)建類

 方海龍的書館 2015-03-03
 深入淺出Cocoa 之動態(tài)創(chuàng)建類
CC許可,,轉(zhuǎn)載請注明出處

在前文《深入淺出Cocoa之類與對象》一文中,我已經(jīng)詳細(xì)介紹了ObjC中的 Class 與 Object 的概念,,今天我們來如何在運行
動態(tài)創(chuàng)建類,。下面這個函數(shù)就是應(yīng)用前面講到的Class,MetaClass的概念,,在運行時動態(tài)創(chuàng)建一個類,。這個函數(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,,因為 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 都指向同一個類名字符串
    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 注冊到運行時系統(tǒng)中
    objc_addClass( new_class );
    
    return YES;
}

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

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

在實際的運用中,,我們使用 ObjC 運行時函數(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,,并在運行時系統(tǒng)中注冊,這樣我們就可以使用這個新類了,。在這里使用 performSelector 來向新類的對象發(fā)送消息,,可以避免編譯警告信息(因為我們并沒有聲明該類及其可響應(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 是其自身


    本站是提供個人知識管理的網(wǎng)絡(luò)存儲空間,,所有內(nèi)容均由用戶發(fā)布,,不代表本站觀點。請注意甄別內(nèi)容中的聯(lián)系方式,、誘導(dǎo)購買等信息,,謹(jǐn)防詐騙。如發(fā)現(xiàn)有害或侵權(quán)內(nèi)容,,請點擊一鍵舉報,。
    轉(zhuǎn)藏 分享 獻(xiàn)花(0

    0條評論

    發(fā)表

    請遵守用戶 評論公約

    類似文章 更多