runtime-分类为什么不生成setter和getter
前言
前几天有人问我一个问题:为什么分类不能自动创建get set方法。老实说,笔者从来没有去思考过这个问题。于是这次通过代码实践跟runtime
源码来探究这个问题。
准备工作
为了能减少输出类数据的代码工作,笔者基于NSObject
的分类封装了一套代码
其中输出类实例变量的具体代码:
- (void)logIvarsWithExpReg: (NSString *)expReg customed: (BOOL)customed {
[NSObject kRecordOBJ];
unsigned int ivarCount;
Ivar * ivars = class_copyIvarList([self class], &ivarCount);
for (int idx = 0; idx < ivarCount; idx++) {
Ivar ivar = ivars[idx];
NSString * ivarName = [NSString stringWithUTF8String: ivar_getName(ivar)];
if (customed && [kOBJIvarNames containsObject: ivarName]) {
continue;
}
if (expReg && !kValidExpReg(ivarName, expReg)) {
continue;
}
printf("ivar: %s --- %s\n", NSStringFromClass([self class]).UTF8String, ivarName.UTF8String);
}
free(ivars);
}
+(void)kRecordOBJ
采用dispatch_once
的方式将NSObject
存在的数据存储到三个数组中,用来排除父类的数据输出
类的属性
正常创建类
@interface Person: NSObject {
int _pId;
} @property (nonatomic, copy) NSString * name;
@property (nonatomic, assign) NSUInteger age; @end int main(int argc, char * argv[]) {
@autoreleasepool {
Person * p = [[Person alloc] init];
[p logCustomIvars];
[p logCustomMethods];
[p logCustomProperties];
return UIApplicationMain(argc, argv, nil, NSStringFromClass([AppDelegate class]));
}
}运行结果:属性
name
和age
生成了对应的_propertyName
的实例变量以及setter
和getter
动态生成属性
age
@implementation Person
@dynamic age; @end运行结果:缺少了
_age
变量以及对应的setAge:
和age
方法手动实现
setter/getter
@implemetation Person
@dynamic age; - (void)setAge: (NSUInteger)age {}
- (NSUInteger)age { return 18; } @end输出结果:未生成
_age
实例变量手动实现
_pId
的setter/getter
@implemetation Person
@dynamic age; - (void)setAge: (NSUInteger)age {}
- (NSUInteger)age { return 18; } - (void)setPId: (int)pId { _pId = pId; }
- (int)pId { return _pId; } @end [p setValueForKey: @"pId"];运行结果:
KVC
的访问会触发setter
方法,_pId
除了无法通过点语法访问外,其他表现与@property
无异
通过上面的几段试验,可以得出@property
的公式:
分类属性
分类中添加
weigh
和height
属性@interface Person (category) @property (nonatomic, assign) CGFloat weigh;
@property (nonatomic, assign) CGFloat height; @end运行结果:
weigh
和height
未生成实例变量以及对应的setter/getter
,与@dynamic
修饰的age
表现一致使用
@synthesize
自动合成setter/getter
方法时编译报错手动实现
setter/getter
@implemetation Person (category)- (void)setWeigh: (CGFloat)weigh {}
- (CGFloat)weigh { return 150; } @end运行结果:与
@dynamic age
后重写其setter/getter
表现一致动态绑定属性来实现
setter/getter
void * kHeightKey = &kHeightKey;
@implemetation Person (category) - (void)setWeigh: (CGFloat)weigh {}
- (CGFloat)weigh { return 150; } - (void)setHeight: (CGFloat)height {
objc_setAssociatedObject(self, kHeightKey, @(height), OBJC_ASSOCIATION_RETAIN_NONATOMIC);
}
- (CGFloat)height {
return return [objc_getAssociatedObject(self, kHeightKey) doubleValue];;
} @end [p logCustomIvars]
[p logCustomMethods];
[p logCustomProperties]; CGFloat height = 180;
p.height = 180;
height = p.height; [p logCustomIvars]
[p logCustomMethods];
[p logCustomProperties];运行结果:动态绑定前后
ivar
没有发生任何变化
通过代码实验,可以得出下面两个结论:
- 分类属性相当于
@dynamic property
- 缺少
ivar
的情况下无法使用@synthesize
自动合成属性
以及一个猜想:
- 在类完成加载后无法继续添加
ivar
通过runtime动态创建类验证猜想:
int main(int argc, char * argv[]) {
NSString * className = @"Custom";
Class customClass = objc_allocateClassPair([NSObject class], className.UTF8String, 0);
class_addIvar(customClass, @"ivar1".UTF8String, sizeof(NSString *), 0, "@");
objc_property_attribute_t type1 = { "T", "@\"NSString\"" };
objc_property_attribute_t ownership1 = { "C", "N" };
objc_property_attribute_t atts1[] = { type1, ownership1 };
class_addProperty(customClass, "property1", atts1, 2);
objc_registerClassPair(customClass);
id instance = [[customClass alloc] init];
NSLog(@"\nLog Ivars ===================");
[instance logCustomIvars];
NSLog(@"\nLog methods ===================");
[instance logCustomMethods];
NSLog(@"\nLog properties ===================");
[instance logCustomProperties];
class_addIvar(customClass, @"ivar2".UTF8String, sizeof(NSString *), 0, "@");
objc_property_attribute_t type2 = { "T", "@\"NSString\"" };
objc_property_attribute_t ownership2 = { "C", "N" };
objc_property_attribute_t atts2[] = { type2, ownership2 };
class_addProperty(customClass, "property2", atts2, 2);
instance = [[customClass alloc] init];
NSLog(@"\nLog Ivars ===================");
[instance logCustomIvars];
NSLog(@"\nLog methods ===================");
[instance logCustomMethods];
NSLog(@"\nLog properties ===================");
[instance logCustomProperties];
}
运行结果:在调用class_registerClassPair
后,添加ivar
失败
从源码解析
objc_class
的结构体定义如下:
struct objc_class : objc_object {
Class superclass;
const char *name;
uint32_t version;
uint32_t info;
uint32_t instance_size;
struct old_ivar_list *ivars;
struct old_method_list **methodLists;
Cache cache;
struct old_protocol_list *protocols;
// CLS_EXT only
const uint8_t *ivar_layout;
struct old_class_ext *ext;
}
ps: 在新版本中结构体内部已经发生了大改,但是内部的属性大致上仍是这些
这里面有个重要的属性ivar_layout
,顾名思义存放的是变量的位置属性,与之对应的还有一个weakIvarLayout
变量,不过在默认结构中没有出现。这两个属性用来记录ivar
哪些是strong
或者weak
,而这个记录操作在runtime
阶段已经被确定好。正由于如此,这极有可能是ivar
无法在类被加载后继续添加的原因之一。ivar_layout
的更多了解可以参照Objective-C Class Ivar layout一文
import
操作帮助编译检查和链接过程,但是在category
的加载过程中,不会将扩展的内容添加到原始的类结构中。runtime
对于category
的加载过程可以简单的分成下面几步(摘自objc category的密码):
objc runtime
的加载入口是一个叫_objc_init
的方法,在library
加载前由libSystem dyld
调用,进行初始化操作- 调用
map_images
方法将文件中的image
map
到内存 - 调用
_read_images
方法初始化map
后的image
,这里面干了很多的事情,像load
所有的类、协议和category
,著名的+ load
方法就是这一步调用的
-仔细看category
的初始化,循环调用了_getObjc2CategoryList
方法,这个方法拿出来看看: - .…
这一切的过程发生在_objc_init
函数中,函数实现如下
简单来说在load_images
函数中最终会走到下面的代码调用来加载所有的类以及类的分类
根据上面的代码加上runtime
的加载顺序,可以继续推出:
@dynamic
实际上是将属性的加载推迟到类加载完成后
另外,前面也说过在缺少ivar
的情况下无法自动合成setter/getter
,除了category
本身是不被添加到类结构中的,所以无法使用类结构的ivar
合成属性外,还有分类自身结构的问题
struct category_t {
const char *name; /// 类名
classref_t cls; /// 类指针
struct method_list_t *instanceMethods; /// 实例方法
struct method_list_t *classMethods; /// 类方法
struct protocol_list_t *protocols; /// 扩展的协议
struct property_list_t *instanceProperties; /// 扩展属性
method_list_t *methodsForMeta(bool isMeta) { ... }
property_list_t *propertiesForMeta(bool isMeta) { ... }
};
可以看到分类结构本身是不存在ivar
的容器的,因此缺少了自动合成属性的条件。最后还有一个问题,我们在使用objc_associate
系列函数绑定属性的时候这些变量存储在了哪里?
总结
首先,iOS的分类在runtime
实现的结构体中并不存在Ivar
类型的容器,缺少了自动合成setter
以及getter
的必要条件,因此在分类中声明的属性默认为@dynamic
修饰。
其次,OC本身是一门原型语言,对象和类原型很像。类对象执行alloc
方法就像是原型模式中的copy
操作一样,类保存了copy
所需的实例信息,这些信息内存信息在runtime
加载时就被固定了,没有扩充Ivar
的条件。(感谢大表哥的科普)
最后,在runtime
中存在一个类型为AssociationHashMap
的哈希映射表保存着对象动态添加的属性,每个对象以自身地址为key
维护着一个绑定属性表,我们动态添加的属性就都存储在这个表里,这也是动态添加property
能成功的基础。
作者:sindri的小巢
链接:http://www.jianshu.com/p/dcc3284b65bf
來源:简书
著作权归作者所有。商业转载请联系作者获得授权,非商业转载请注明出处。
runtime-分类为什么不生成setter和getter的更多相关文章
- 【原】iOS 同时重写setter和getter时候报错:Use of undeclared identifier '_name';did you mean 'name'
写了那么多的代码了,平时也没有怎么注意会报这个错误,因为平时都很少同时重写setter和getter方法,一般的话,我们大概都是使用懒加载方法,然后重写getter方法,做一个非空判断.然后有时候根据 ...
- 如果将synthesize省略,语义特性声明为assign retain copy时,自己实现setter和getter方法
如果将synthesize省略,并且我们自己实现setter和getter方法时,系统就不会生成对应的setter和getter方法,还有实例变量 1,当把语义特性声明为assign时,setter和 ...
- 假设将synthesize省略,语义特性声明为assign retain copy时,自己实现setter和getter方法
假设将synthesize省略,而且我们自己实现setter和getter方法时,系统就不会生成相应的setter和getter方法,还有实例变量 1,当把语义特性声明为assign时,setter和 ...
- 假设synthesize省略,语义属性声明assign retain copy时间,为了实现自己的setter和getter方法
假设synthesize省略,而且我们自己实现setter和getter方法时,系统就不会生成相应的setter和getter方法,还有实例变量 1,当把语义特性声明为assign时,setter和g ...
- Lombok的@Data、@Setter、@Getter注解没反应问题解决
在用@Data注解时,没有生成setter/getter方法.百度了一堆都没解决方法,后来用Google查了一下解决了~~~ 使用IDEA需要安装Lombok插件,我这里已经下载好,如果没下载安装点击 ...
- 自定义类属性设置及setter、getter方法的内部实现
属性是可以说是面向对象语言中封装的一个体现,在自定义类中设置属性就相当于定义了一个私有变量.设置器(setter方法)以及访问器(getter方法),其中无论是变量的定义,方法的声明和实现都是系统自动 ...
- iOS开发核心语言Objective C —— 面向对象思维、setter和getter方法及点语法
本分享是面向有意向从事iOS开发的伙伴们.或者已经从事了iOS的开发人员.假设您对iOS开发有极高的兴趣,能够与我一起探讨iOS开发.一起学习,共同进步.假设您是零基础,建议您先翻阅我之前分享的iOS ...
- setter 和 getter 高级 以及内存管理初级
setter 和 getter 的演变,紧接setter 和 getter 初级 1.@property 和 @synthesize 这两个关键字的出现,就是为了剔除代码中的setter方法和get ...
- 关于setter 和 getter方法的一些总结(初级)
1.最基础的set 和 get 准备工作 Person.h @interface Person : NSObject { NSString *_hobby; // ObjC建议成员变量带"_ ...
随机推荐
- json字符串转为json对象-jQuery.parseJSON()
<!DOCTYPE html> <html> <head> <meta charset="UTF-8"> <title> ...
- Delphi中定义了四种布尔类型:Boolean,ByteBool,WordBool和LongBool。后面三种布尔类型是为了与其他语言兼容而引入的
bool是LongBool类型. Delphi中定义了四种布尔类型:Boolean,ByteBool,WordBool和LongBool.后面三种布尔类型是为了与其他语言兼容而引入的,一般情况下建议使 ...
- jQuery的$.fn使用
jquery中文网为您提供jQuery的$.fn使用等资源,欢迎您收藏本站,我们将为您提供最新的jQuery的$.fn使用资源 $.fn是指jquery的命名空间,加上fn上的方法及属性,会对jque ...
- C#的三大难点之二:托管与非托管
相关文章: C#的三大难点之前传:什么时候应该使用C#?C#的三大难点之一:byte与char,string与StringBuilderC#的三大难点之二:托管与非托管C#的三大难点之三:消息与事件 ...
- CSS实现绝对定位居中
我们经常用margin:0 auto来实现水平居中,而一直认为margin:auto不能实现垂直居中……实际上,实现垂直居中仅需要声明元素高度和下面的CSS: .Absolute-Center { m ...
- 基于java spring框架开发部标1078视频监控平台精华文章索引
部标1078视频监控平台,是一个庞杂的工程,涵盖了多层协议,部标jt808,jt809,jt1078,苏标Adas协议等,多个平台功能标准,部标796标准,部标1077标准和苏标主动安全标准,视频方面 ...
- Debian NAT共享上网
如果Linux主机有两个网卡,比如一个有线.一个无线,当无线连接后,其他机器即可通过有线共享上网,为了方便叙述,假设环境如下: A机器有两块网卡,eth0和ws0,其中ws0为无线网卡,已连接wifi ...
- kafka官方Quick Start
1.下载kafka,并上传到服务器 2.如果之前没安装zookeeper,这里可以启动一个简单的zookeeper bin/zookeeper-server-start.sh config/zooke ...
- GreenDao学习
官方文档地址:http://greenrobot.org/greendao/documentation//introduction/ 英语不好的可以使用谷歌翻译,很轻松的看懂文档,不需要FQ. 看不懂 ...
- Oracle关于快速缓存区应用原理
为什么oracle可以对于大量数据进行訪问时候能彰显出更加出色表现,就是通过所谓的快速缓存来实现数据的快速运算与操作.在之前的博文中我已经说过sql的运行原理,当我们訪问数据库的数据时候,首先不是从数 ...