如何访问并修改一个类的私有属性?
三种方式供你选择
- 通过KVC来设置
- 通过runtime动态改变
- 通过 msg_send() 设置
先定义这样一个类 PrivateVariablesClass:
#import "PrivateVariablesClass.h"
@interface PrivateVariablesClass ()
@property (nonatomic, assign) NSInteger priviteNum;
@property (nonatomic, strong) UIView *priviteView;
@end
@implementation PrivateVariablesClass
-(void)showPropertyPrivateVariablesClass
{
    NSLog(@"priviteNum = %@", @(_priviteNum));
    NSLog(@"priviteView = %@", _priviteView);
}
@end
我们在外部改完后,调用 showPropertyPrivateVariablesClass 方法去查看这两个私有属性的值。
1.KVC:
-(void)way1{
    PrivateVariablesClass *classA = [[PrivateVariablesClass alloc] init];
    [classA setValue:@(4) forKey:@"_priviteNum"];
    [classA setValue:self.view forKey:@"_priviteView"];
    [classA showPropertyPrivateVariablesClass];
}
结果输出:
priviteNum = 4  
priviteView = <UIView: 0x7fae50f07e10; frame = (0 0; 375 667); autoresize = W+H; layer = <CALayer: 0x600003c08020>>
2.runtime动态改变
-(void)way2{
    PrivateVariablesClass *classA = [[PrivateVariablesClass alloc] init];
    
    unsigned int outCount = 0;
    Ivar *ivars = class_copyIvarList([PrivateVariablesClass class], &outCount);
    
    for (int i = 0; i < outCount; i ++) {
        Ivar ivar = ivars[i];
        
        const char *ivarName = ivar_getName(ivar);
        
        //这里要注意ARC下, 这个会报错
        /**
         在修改NSInteger型变量的时候,ARC下,编译器不允许你将NSInteger类型的值赋值给id,在buildsetting中将Objective-C Automatic Reference Counting修改为No即可。但是这样工程就会变成MRC,所以,如果是非对象类型就不建议用object_setIvar这样的方法去修改了。
         */
 
        int a = strcmp(ivarName, "_priviteNum");
        if (strcmp(ivarName, "_priviteNum") == 0) {
            //这种方式传值int类型会报错,不能传入
            object_setIvar(classA, ivar, 22);
        }
        
        if (strcmp(ivarName, "_priviteView") == 0) {
            object_setIvar(classA, ivar, self.view);
        }
    }
    
    [classA showPropertyPrivateVariablesClass];
}
结果输出:
priviteNum = 22
priviteView = <UIView: 0x7ffb08e0b040; frame = (0 0; 375 667); autoresize = W+H; layer = <CALayer: 0x60000398ad40>>
3.msg_send()
既然是私有属性了,必然有setter方法, 那我们动态调用一下(适用私有属性,不适用私有变量)。
-(void)way3{
    PrivateVariablesClass *classA = [[PrivateVariablesClass alloc] init];
    ((void (*)(id, SEL, int))(void *) objc_msgSend)((id)classA, @selector(setPriviteNum:) , 33);
    ((void (*)(id, SEL, id))(void *) objc_msgSend)((id)classA, @selector(setPriviteView:) , self.view);
    [classA showPropertyPrivateVariablesClass];
}
结果输出:
priviteNum = 33
priviteView = <UIView: 0x7fd4bfa079f0; frame = (0 0; 375 667); autoresize = W+H; layer = <CALayer: 0x600001085f40>>
 
 
 
 
