runtime作为Objective-C 运行时机制,是其一项核心技术,下面列举我们常见的一些用法。
1.动态修改变量
/** 1.动态修改变量(私有变量也可修改)【常用】*/
- (void)dynamicModifyVariable {
// Swift的Person类的变量类型需为继承NSObject的类
NSLog(@"修改前姓名为:%@", myPeople.name);
unsigned int count = 0;
// 获取类的成员变量列表(包括私有) 获取属性,方法,协议列表 类似
Ivar *varList = class_copyIvarList([People class], &count);
for (int i = 0; i < count; i++) {
Ivar var = varList[i];
const char *varName = ivar_getName(var);
NSString *proname = [NSString stringWithUTF8String:varName];
// Person 为Swift类的话,若name为属性,则为"name"
if ([proname isEqualToString:@"_name"]) {
object_setIvar(myPeople, var, @"xiaoDong");
break;
}
}
NSLog(@"1.修改后姓名为:%@",myPeople.name);
}
People 模型类中
- (instancetype)init {
if (self = [super init]) {
_name = @"xiaoBao";
_sex = @"man";
}
return self;
}
2.动态添加方法
常用于不修改开源库源代码的基础上,给其添加方法,并且可使用以及修改开源库中的私有变量;类别也可实现,开源库中私有变量无法使用
/** 2.动态添加方法 【常用于给开源库添加方法 类别也可实现】*/
- (void)dynamicAddMethod {
// 注册一个方法
SEL getInformationSelector = sel_registerName("getPersonAllInfo");
/* 将函数指针指向方法 IMP:指向实际执行函数体的指针 type函数返回值和参数类型
"v@:" v代表无返回值void,如果是i则代表int;@代表id; :代表SEL _cmd; */
class_addMethod([People class], getInformationSelector, (IMP)getInformation, "v@");
// 测试
[self testPersonAddMentod];
}
// C 函数
void getInformation(id aPerson) {
People *pp = (People *)aPerson;
NSLog(@"2.添加的方法,获取全部信息:%@,%@",pp.name,pp.sex);
}
/** 测试Person类新添加的方法 */
- (void)testPersonAddMentod {
SEL getPersonAllInfo = sel_registerName("getPersonAllInfo");
// 若调用新增方法 则调用函数
if ([myPeople respondsToSelector:getPersonAllInfo]) {
getInformation(myPeople);
}
}
3.动态交换方法
/** 3.动态交换方法 */
- (void)dynamicExchangeMethod {
// class_getInstanceMethod 获取对象方法, class_getClassMethod 获取类方法
Method m1 = class_getInstanceMethod([People class], @selector(getPersonName));
Method m2 = class_getInstanceMethod([People class], @selector(getPersonSex));
method_exchangeImplementations(m1, m2);
NSLog(@"3.交换方法后:%@,%@",[myPeople getPersonName], [myPeople getPersonSex]);
}
People 模型类中
#pragma mark - public
- (NSString *)getPersonName {
return _name;
}
- (NSString *)getPersonSex {
return _sex;
}
+ (NSString *)getPersonAge {
return @"18";
}
+ (NSString *)getPersonSchool {
return @"BeiDa";
}
4.动态拦截或者替换方法
/** 4.动态拦截或者替换方法【常用于替换开源库的方法】*/
- (void)dynamicReplaceMethod {
// 用本类中的对象方法与Person类中的类方法交换 从而实现替换
Method m1 = class_getClassMethod([People class], @selector(getPersonSchool));
Method m2 = class_getInstanceMethod([ViewController class], @selector(replaceMethodTest1));
method_exchangeImplementations(m1, m2);
// 用下面方法也可 直接替换
//method_setImplementation(m1, (IMP)replaceMethodTest2);
[People getPersonSchool];
}
/** 替换方法测试 */
- (void)replaceMethodTest1 {
NSLog(@"4.替换方法成功");
}
void replaceMethodTest2() {
NSLog(@"4.替换方法成功");
}
5.动态添加属性
/** 5.动态添加属性 */
- (void)dynamicAddProperty {
// 在Person的扩展类中设置关联
myPeople.telephone = @"15688886666";
NSLog(@"5.添加的属性:%@",myPeople.telephone);
}
@implementation People (AddProperty)
// 重写set和get方法 设置关联
- (NSString *)telephone {
return objc_getAssociatedObject(self, "telephone");
}
- (void)setTelephone:(NSString *)telephone {
// OBJC_ASSOCIATION_RETAIN_NONATOMIC 为关联策略
objc_setAssociatedObject(self, "telephone", telephone, OBJC_ASSOCIATION_RETAIN_NONATOMIC);
}
6.自动归档和解档
/** 6.自动归档和解档*/
- (void)automaticArchiveAndUnarchive {
// 见People类中的encodeWithCoder和initWithCoder实现
NSLog(@"自动归档和解档成功");
}
#pragma mark - NSCoding
// 可以将这两个方法内代码写成全局宏或者方法 然后一行代码调用即可
- (void)encodeWithCoder:(NSCoder *)aCoder {
unsigned int count = 0;
Ivar *ivarList = class_copyIvarList([People class], &count);
for (int i = 0; i < count; i ++) {
Ivar ivar = ivarList[i]; // 从成员列表中取出成员变量
const char *name = ivar_getName(ivar); // 获取成员变量名
// 进行归档
NSString *key = [NSString stringWithUTF8String:name];
id value = [self valueForKey:key];
[aCoder encodeObject:value forKey:key];
}
free(ivarList);
//ENCODE_RUNTIME(People) // 若写成宏 调用
}
- (id)initWithCoder:(NSCoder *)aDecoder {
if (self = [super init]) {
unsigned int count = 0;
Ivar *ivarList = class_copyIvarList([People class], &count);
for (int i = 0; i < count; i++) {
Ivar ivar = ivarList[i]; // 从成员列表中取出成员变量
const char *name = ivar_getName(ivar); // 获取成员变量名
// 进行解档
NSString *key = [NSString stringWithUTF8String:name];
id value = [aDecoder decodeObjectForKey:key];
// 将值赋值给成员变量
[self setValue:value forKey:key];
}
free(ivarList);
}
return self;
//DECODE_RUNTIME(People) // 若写成宏 调用
}
7.字典转模型
/** 7.字典转模型 */
- (void)modelConvertFromDictionary {
NSDictionary *dic = @{@"peoplename":@"hello", @"sex":@"unknown",@"books":@[@"math",@"english",@"history"],@"dog":@{@"dogname":@"wangcai",@"dogage":@"1"}};
// 若dic的key较多 可用下面方法自动打印出模型属性代码 然后拷贝使用即可
[dic propertyLog];
myPeople = [People modelWithDictionary:dic];
NSLog(@"字典转模型后:%@,%@,%@",myPeople.name, myPeople.sex, myPeople.books);
}
@implementation NSDictionary (PropertyLog)
- (void)propertyLog {
NSMutableString *properties = [[NSMutableString alloc] init];
// 遍历字典
[self enumerateKeysAndObjectsUsingBlock:^(id _Nonnull key, id _Nonnull obj, BOOL * _Nonnull stop) {
NSString *property;
if ([obj isKindOfClass:[NSString class]]) {
property = [NSString stringWithFormat:@"@property (nonatomic, copy) NSString *%@;", key];
} else if ([obj isKindOfClass:[NSArray class]]) {
property = [NSString stringWithFormat:@"@property (nonatomic, strong) NSArray *%@;", key];
} else if ([obj isKindOfClass:[NSDictionary class]]) {
property = [NSString stringWithFormat:@"@property (nonatomic, strong) NSDictionary *%@;", key];
} else if ([obj isKindOfClass:[NSNumber class]]) {
property = [NSString stringWithFormat:@"@property (nonatomic, assign) NSInteger %@;", key];
}
[properties appendFormat:@"\n%@\n", property];
}];
NSLog(@"%@", properties);
}
People 模型类中
/** 字典转模型 */
+ (instancetype)modelWithDictionary:(NSDictionary *)dict {
return [self modelWithDictionary:dict modelClass:NSStringFromClass([self class])];
// MODEL_WITH_DICTIONARY(dict, NSStringFromClass([self class])) // 若写成宏 调用
}
// 下面方法可写在BaseModel中, 这样Model中直接调用即可 也可以写成全局宏
+ (instancetype)modelWithDictionary:(NSDictionary *)dict modelClass:(NSString *)modelClass {
Class ModelClass = NSClassFromString(modelClass);
id model = [[ModelClass alloc] init];
/* 1.KVC方式(推荐) 将dict的key对应的值赋值给对应的model对应的属性
必须保证model中的属性和dict中的key一一对应 */
[model setValuesForKeysWithDictionary:dict];
// 2.runTime方式 较为繁琐(不推荐)如果字典中有字典,字典中有数组,需要多级转换
return model;
}
#pragma mark - override
// 防止model中没有key对应的属性 造成崩溃
- (void)setValue:(id)value forUndefinedKey:(NSString *)key {
NSLog(@"UndefineKey = %@",key);
// 在此也可实现 字典中的key比model中的属性还多的情况, 比如
if ([key isEqualToString:@"peoplename"]) {
_name = value;
}
}