美文网首页程序员IOSiOS Developer
Objective-C的运行时常用方法

Objective-C的运行时常用方法

作者: luomagaoshou | 来源:发表于2016-08-29 00:09 被阅读268次

    前言

    runtime是OC成为面向对象的基础,了解runtime的基础用法对理解OC非常必要。大量的开源库,如MJExtension,就是使用运行时动态获取类的属性并实现动态赋值。熟悉runtime的方法也是看开源库的基础之一。

    一、OC方法篇

    OC对象的方法调用是利用selector寻找对应的imp指针进行调用。
    现在我们在代码中理解原理。
    定义三个函数,该方法名为imp指针

    void testFunc1(id self, SEL _cmd) {
    
    NSLog(@"实现testFunc1");
    
    }
    void testFunc2(id self, SEL _cmd) {
        
        NSLog(@"实现testFunc2");
        
    }
    void testFunc3(id self, SEL _cmd) {
        
        NSLog(@"实现testFunc3");
        
    }
    

    添加实例方法

    //  添加imp到sel,该方法返回一个bool值,成功为YES
        BOOL isAddInstanceMethodFormImpSuccess = class_addMethod(class, sel_registerName("instanceMethod"), (IMP)testFunc1, "v@:");
        if (isAddInstanceMethodFormImpSuccess) {
            NSLog(@"添加imp方法成功");
        }
    

    添加类方法

        BOOL isAddClassMethodFormImpSuccess = class_addMethod(object_getClass(class), sel_registerName("classMethod"), (IMP)testFunc3, "v@:");
        if (isAddClassMethodFormImpSuccess) {
            NSLog(@"添加imp方法成功");
        }
    

    取得blcok的imp指针

      IMP impOfBlock = imp_implementationWithBlock(^(id obj, NSString *string,...){
          NSLog(@"调用者:%@\nstring:%@", obj, string);
      });
      ```
      添加impOfBlock到sel
      BOOL isAddMethodFormImpOfBlockSuccess = class_addMethod(class, sel_registerName("instanceMethodImpOfBlock:"), impOfBlock, "v@:@");
      if (isAddMethodFormImpOfBlockSuccess) {
          NSLog(@"添加impOfBlock方法成功");
      }
      ```
    
    添加方法后,由于这是运行时添加的,我们不能像平时那样调用,要用使用performSelector系列方法进行调用
    

    [object performSelector:@selector(instanceMethod)];
    [object performSelector:@selector(instanceMethodImpOfBlock:) withObject:@"我要调用Block"];
    [(id)class performSelector:@selector(classMethod)];

    使用UIView作为测试类,该viewController的view作为测试对象,该结果为
    

    2016-08-29 00:20:52.404 RuntimeDemo[3515:1637694] 实现testFunc1
    2016-08-29 00:20:52.404 RuntimeDemo[3515:1637694] 调用者:<UIView: 0x7ffbdbb04f80; frame = (0 0; 414 736); autoresize = W+H; layer = <CALayer: 0x7ffbdbb03580>>
    string:我要调用Block
    2016-08-29 00:20:52.405 RuntimeDemo[3515:1637694] 实现testFunc3

    
    ##二,成员变量与属性篇
    由于在OC在只有在类的创建期间才能添加ivar,但是我们启动app时,类都创建好了,无法添加,所以我在在运行期间自己创建一个类
    

    根据名字取得类
    Class testClass = objc_getClass("MLView");
    if (!testClass) {
    //动态创建类
    testClass = objc_allocateClassPair([UIView class], "MLView", 0);
    }

    添加ivar
    

    BOOL isAddIvarSuccess = class_addIvar(class, "_runtimeString", sizeof(NSString *), log(sizeof(NSString *)), "@"NSString"");
    if (isAddIvarSuccess) {
    NSLog(@"添加ivar成功");
    }

    添加property
    
    objc_property_attribute_t type = { "T", "@\"NSString\"" };
     objc_property_attribute_t backingivar  = { "V", "_runtimeString" };
    objc_property_attribute_t attrs[] = {type, backingivar};
    BOOL isAddPropertySuccess = class_addProperty(class, "runtimeString", attrs, sizeof(attrs)/sizeof(objc_property_attribute_t));
    if (isAddPropertySuccess) {
        NSLog(@"添加property成功");
    }
    
     注册类名,并将viewController的view的class设置为运态创建的类(实践上不建议这样做,不易检查错误,如要添加属性可以变相用关联添加,下文用介绍如何在分类添加属性)
    

    objc_registerClassPair(testClass);
    //使用view作为 测试对象
    id testObject =self.view;
    //设置testObject的类,就可以使用动态创建的ivar
    object_setClass(testObject, testClass);

    添加完变量,成功了一半,现在开始测试是否添加成功就要进行赋值操作
    
      获取ivar并赋值
    
    Ivar ivar = class_getInstanceVariable(class, "_runtimeString");
    object_setIvar(object, ivar, @"addIvar测试字符串");
    NSLog(@"addIvar的值:%@", object_getIvar(object, ivar));
    
      获取property并赋值
    
    objc_property_t property = class_getProperty(class, "runtimeString");
    const char *propertyAttr = property_getAttributes(property);
    NSLog(@"addProperty的细节:%s", propertyAttr);
    [object setValue:@"addProperty测试字符串" forKey:@"_runtimeString"];
    NSLog(@"addProperty的值:%@", [object valueForKey:@"_runtimeString"]);
    
    运行后得到的结果为
    

    2016-08-29 00:34:15.870 RuntimeDemo[8155:1663698] addIvar的值:addIvar测试字符串
    2016-08-29 00:34:15.870 RuntimeDemo[8155:1663698] addProperty的细节:T@"NSString",V_runtimeString
    2016-08-29 00:34:18.211 RuntimeDemo[8155:1663698] addProperty的值:addProperty测试字符串

    
    
    ##三、运行时简单封装
    ####1、在分类中添加属性
    现在我要在给NSObject分类里添加一个对象
    

    @property (nonatomic, strong) NSString *featureIdentifier;

    然后重写setter和getter方法,并用关联机制
    
    • (void)setFeatureIdentifier:(NSString *)featureIdentifier
      {
      objc_setAssociatedObject(self, @selector(featureIdentifier), featureIdentifier, OBJC_ASSOCIATION_RETAIN_NONATOMIC);
      }
    • (NSString *)featureIdentifier
      {
      return objc_getAssociatedObject(self, @selector(featureIdentifier));
      }
    ####2、获取该类的ivar列表
    
    • (NSArray *)arrayOfIvars
      {
      unsigned int count = 0;
      Ivar *ivar = class_copyIvarList(self, &count);
      NSMutableArray *ivarNameArray = [[NSMutableArray alloc] init];
      for (int i = 0; i<count; i++) {
      Ivar iva = ivar[i];
      const char *name = ivar_getName(iva);
      NSString *strName = [NSString stringWithUTF8String:name];
      [ivarNameArray addObject:strName];

      }

      free(ivar);
      return ivarNameArray;

    }

    ####3、取得该类属性列表
    
    • (NSArray *)arrayOfProperties
      {
      unsigned int count = 0;
      objc_property_t *properties = class_copyPropertyList(self, &count);

      NSMutableArray *propertyNameArray = [[NSMutableArray alloc] init];
      for (int i = 0; i<count; i++) {
      objc_property_t property = properties[i];
      const char *name = property_getName(property);
      NSString *strName = [NSString stringWithUTF8String:name];
      [propertyNameArray addObject:strName];

      }

      free(properties);
      return propertyNameArray;

    }

    
    ####4、取得该类实例方法列表
    
    • (NSArray *)arrayOfInstanceMethods
      {
      unsigned int count = 0;
      Method *methodList = class_copyMethodList(self, &count);
      NSMutableArray *methods = [[NSMutableArray alloc] init];

      for (NSInteger i = 0; i < count; i++) {
      SEL selector = method_getName(methodList[i]);
      NSString *selString = NSStringFromSelector(selector);
      [methods addObject:selString];
      }

      free(methodList);
      return methods;
      }

    ####5、取得该类方法方法列表
    
    unsigned int count = 0;
    Method *methodList = class_copyMethodList(object_getClass(self), &count);
    NSMutableArray *methods = [[NSMutableArray alloc] init];
    
    for (NSInteger i = 0; i < count; i++) {
        SEL selector = method_getName(methodList[i]);
        NSString *methodString = NSStringFromSelector(selector);
        [methods addObject:methodString];
    }
    
    free(methodList);
    return methods;
    
    ####6、取得该类遵循协议列表
    
    • (NSArray *)arrayOfProtocols{

      unsigned int count = 0;
      Protocol * __unsafe_unretained *protocols = class_copyProtocolList(self, &count);
      NSMutableArray *protocolArray = [[NSMutableArray alloc] init];
      for (int i = 0; i < count; i++) {
      [protocolArray addObject:[NSString stringWithUTF8String:protocol_getName(protocols[i])]];

      }

    free(protocols);
    return protocolArray;
    

    }

    ####7、取得该工程所有类的列表
    
    • (NSArray *)arrayOfAllClass {
      NSMutableArray * classList = [NSMutableArray array];

      unsigned int classesCount = 0;
      Class * classes = objc_copyClassList(&classesCount);

    for ( int i = 0; i < classesCount; ++i) {
        NSString *className = NSStringFromClass(classes[i]);
    
        [classList addObject:className];
    }
    
    free(classes);
    return classList;
    

    }

    ####8、取得该对象中有值的的property字典
    
    • (NSDictionary *)dictionaryOfPropertyKeyValues
      {
      NSArray *properties = [[self class] arrayOfProperties];
      NSMutableDictionary *keyValueDictionary = [[NSMutableDictionary alloc] init];
      for (NSInteger i = 0; i < properties.count; i++) {
      if ([self valueForKey:properties[i]]) {
      [keyValueDictionary setObject:[self valueForKey:properties[i]] forKey:properties[i]];
      }
      }
      return keyValueDictionary;
      }
    
    [此处应该有demo][https://github.com/luomagaoshou/MLRuntimeDemo]

    相关文章

      网友评论

        本文标题:Objective-C的运行时常用方法

        本文链接:https://www.haomeiwen.com/subject/tcdxettx.html