在前文《深入浅出Cocoa之类与对象》一文中,我已经详细介绍了ObjC中的 Class 与 Object 的概念,今天我们来如何在运行

时动态创建类。下面这个函数就是应用前面讲到的Class,MetaClass的概念,在运行时动态创建一个类。这个函数来自《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;
} // 确保要创建的类不存在
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 分配内存
new_class = calloc( 2, sizeof(struct objc_class) );
meta_class = &new_class[1]; // 设置 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,设置类的 super class
// 2,设置 meta class 的 super class
// 3,设置 meta class 的 isa
new_class->super_class = super_class;
meta_class->super_class = super_class->isa;
meta_class->isa = (void *)root_class->isa; // 最后,将 class 注册到运行时系统中
objc_addClass( new_class ); return YES;
}

如果要在代码中使用运行时相关的函数,我们需要导入 libobjc.dylib,并导入相关的头文件(比如这里的 runtime.h)。

前文中总结到“ObjC 为每个类的定义生成两个 objc_class ,一个即普通的 class,另一个即 metaclass。我们可以在运行期创建这两个 objc_class 数据结构,然后使用 objc_addClass 动态地创建新的类定义。”,这在上面的代码中就体现出来了:new_class 和 meta_class 就是新类所必须的两个 objc_class。其他的代码就不解释了,注释以及代码足以自明了。

在实际的运用中,我们使用 ObjC 运行时函数来动态创建类:

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;
}

在上面的代码中,我们创建继承自 NSString 的子类 NSStringSubclass,然后向其中添加方法 report,并在运行时系统中注册,这样我们就可以使用这个新类了。在这里使用 performSelector 来向新类的对象发送消息,可以避免编译警告信息(因为我们并没有声明该类及其可响应的消息)。

执行结果为:

 >> 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

根据前文中的类关系图,我们不难从执行结果中分析出 NSStringSubclass 的内部类结构:

1,对象的地址为 :0x100114710
2,class 的地址为:0x100114410
3,meta class 的地址为:0x100114560
4,meta class 的 class 地址为:0x7fff7e257b50 (也是 NSObject 的 meta class)
5,NSObject 的 meta class 的 meta class 是其自身

深入浅出Cocoa 之动态创建类【转】的更多相关文章

  1. ios动态创建类Class

    [Objective-C Runtime动态加载]---动态创建类Class 动态创建类Class,动态添加Class成员变量与成员函数,动态变量赋值与取值,动态函数调用等方法 a.使用objc_al ...

  2. python动态创建类的声明

    动态创建类的声明 使用内置函数type,原型:class type(name, bases, dict)name是类的名字,相当于__class__bases是类的基类,元组,可以有多个基类,但是基类 ...

  3. OC 反射-->动态创建类

    系统方法 NSLog(@"%s", __func__); //打印出类的方法名称,如: //打印结果:2018-02-22 10:52:15.394575+0800 DemoRun ...

  4. 使用python type动态创建类

    使用python type动态创建类 X = type('X', (object,), dict(a=1))  # 产生一个新的类型 X 和下列方法class X(object):    a = 1效 ...

  5. Python中type()详解:动态创建类

    众所周知: type()函数可以查看变量的类型: 先看一个简单的列子来看一下type查看变量类型 class Animal(): pass a=Animal() print(type(a)) prin ...

  6. python-获取类名和方法名,动态创建类和方法及属性

    获取类名和方法名1.在函数外部获取函数名称,用.__name__获取2.在函数内部获取当前函数名称,用sys._getframe().f_code.co_name方法获取3.使用inspect模块动态 ...

  7. type动态创建类

    在一些特定场合,需要动态创建类,比如创建表单,就会用到type动态创建类,举个例子: class Person(object): def __init__(self,name,age): self.n ...

  8. 动态创建类/ swizzle class

    动态创建类 Class subclass = objc_allocateClassPair(baseClass, subclassName, );//生成,指定父类 //添加方法,变量...一些操作 ...

  9. Python 中使用动态创建类属性的机制实现接口之后的依赖

    我们在自动化测试中经常会需要关联用例处理,需要动态类属性: 推荐使用第二种方法: 创建:setattr() 获取:getattr() 两种,如何创建 类属性 loan_id # 第一种,创建 # 类名 ...

随机推荐

  1. GULP入门(一)

    1.首先要先装node.然后在命令行里安装全局的gulp: npm install --global gulp 这是gulp在的生成的位置 2.接下来,我们需要将gulp安装到项目本地 npm ins ...

  2. 2018-8-10-WPF-使用不安全代码快速从数组转-WriteableBitmap

    title author date CreateTime categories WPF 使用不安全代码快速从数组转 WriteableBitmap lindexi 2018-08-10 19:16:5 ...

  3. Vue设置element的dialog

    1.设置css:参考https://www.jianshu.com/p/a3eb60b75b92 <style> .el-dialog { max-height: 600px; displ ...

  4. 部分树形DP的优化

    ural1018. Binary Apple Tree 题目大意 有一棵n个节点的树,树上每个节点有一个值,选择m个节点使这些节点值的和最大 要求:如果选当前节点,则必须选它的父节点 解法: 我们设d ...

  5. spring源码学习之bean的加载(三)

    接着二中的继续写,那个都超过1000行了,哈,需要重新写一个,要不太长了,我都看不下去了 7.4 初始化bean doCreateBean函数中有这样一行代码:这行代码中initializeBean函 ...

  6. 洛谷1850(NOIp2016) 换教室——期望dp

    题目:https://www.luogu.org/problemnew/show/P1850 状态里记录的是”上一回有没有申请“,而不是”上一回申请成功否“,不然“申请 j 次”就没法转移了. dou ...

  7. TZ_02MyBatis_一级缓存和二级缓存

    1.Mybatis中的缓存 1>什么是缓存        存在于内存中的临时数据.   2> 为什么使用缓存        减少和数据库的交互次数,提高执行效率.   3>什么样的数 ...

  8. web前端学习(四)JavaScript学习笔记部分(7)-- JavaScript DOM对象控制HTML元素详解

    1.方法 getElementsByName() 获取name 可以获取一个数组类型数据(参数加引号) getElementsByTagName() 获取元素   getAttribute() 获取元 ...

  9. hdu 1505 && hdu1506 &&hdu 2830 && 2870 总结---------DP之状图选最大矩形

    /* 多谢了“闭眼,睁眼” 同学给我一套dp专题,不然真是没接触过这种题型. 做个4个简单的,很高兴有所收获. 2013-08-06 /* HDU 1506 最基础的一道题目,其主要精髓就在于两个数组 ...

  10. 用navicat把MYSQL一张表的数据批追加到另外一张表

    1. 表结构完全一样 insert into 表1 select * from 表22. 表结构不一样(这种情况下得指定列名) insert into 表1 (列名1,列名2,列名3) select ...