博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
runtime之方法的交换
阅读量:4984 次
发布时间:2019-06-12

本文共 2402 字,大约阅读时间需要 8 分钟。

工作中没怎么用到runtime的东西,所以一直没怎么看,现在开始拿起来。

runtime之方法的交换:

都知道OC中有category可以对已知类进行扩展,但是假如工程中需要修改某类的原方法,若用category的话,调用的时候会调用到category中实现的方法,而原方法中的功能就已经被覆盖,这样就调用不到系统的方法了,为了避免这种情况,我们可以用方法交换的形式:

如写一个NSURL的分类,现在将它的URLWithString:的方法替换成我们自己写的方法:

+ (void)load{

  static dispatch_once_t onceToken;
  dispatch_once(&onceToken, ^{
    Method urlWithStringMethod = class_getClassMethod(self, @selector(URLWithString:));

    Method wd_urlWithStringMethod = class_getClassMethod(self, @selector(WD_urlWithString:));

    /*交换了URLWithString和WD_urlWithString的函数指针(该指针指向方法的实现),所以交换了之后,若调用URLWithString:方法,实际上是调用了WD_urlWithString方法的实现,而调用了WD_urlWithString:方法,实际上的调用了URLWithString:方法的实现*/

    method_exchangeImplementations(urlWithStringMethod, wd_urlWithStringMethod);
  });
}

/*

在load中将NSURL中的URLWithString方法跟自定义的WD_urlWithString进行了交换
在外部调用系统的URLWithString:方法,实际上会到WD_urlWithString:中
*/
+ (instancetype)WD_urlWithString:(NSString *)urlString{
  NSURL *url = [NSURL WD_urlWithString:urlString]; //实际调用的URLWithString:方法的实现,所以不会造成无限循环
  if (url == nil) {
    NSLog(@"url is null");
  }
  return url;
}

method:

实际上method是一个结构体,

struct objc_method {

  SEL method_name OBJC2_UNAVAILABLE;
  char *method_types OBJC2_UNAVAILABLE;
  IMP method_imp OBJC2_UNAVAILABLE;
}
method_name 是方法的 selector,可以理解为运行时的方法名;
*method_types 是一个参数和返回值类型编码的字符串;
method_imp 是指向方法实现的指针。
Method Swizzling 的实质是在运行时,访问对象的方法结构体,并改变它的底层实现。

拿上面的例子来说,变化如下:

交换前->

method URLWithString {

  SEL method_name = @selector(URLWithString:)
  char *method_types = “v@:“ //返回值void, 参数id(self),selector(_cmd)
  IMP method_imp = 0x000FFFF //指向([NSURL URLWithString:])
}
method WD_urlWithString {
  SEL method_name = @selector(WD_urlWithString:)
  char *method_types = “v@:“ //返回值void, 参数id(self),selector(_cmd)
  IMP method_imp = 0x000EEEE //指向([NSURL WD_urlWithString:])
}
交换后->
method URLWithString {
  SEL method_name = @selector(URLWithString:)
  char *method_types = “v@:“ //返回值void, 参数id(self),selector(_cmd)
  IMP method_imp = 0x000FFFF //指向([NSURL WD_urlWithString:])
}
method WD_urlWithString {
  SEL method_name = @selector(WD_urlWithString:)
  char *method_types = “v@:“ //返回值void, 参数id(self),selector(_cmd)
  IMP method_imp = 0x000EEEE //指向([NSURL URLWithString:])
}
可以看到使用method_exchangeImplementations实质是交换它们的IMP。
Method Swizzling,我们可以对系统中的方法进行修改,动态添加。
在load中处理,因为 +load 方法会在类被添加到 OC 运行时执行,保证了 Swizzling 方法的及时处理。

转载于:https://www.cnblogs.com/wudan7/p/7422013.html

你可能感兴趣的文章
VUE-es6
查看>>
MySQL-5.7 高阶语法及流程控制
查看>>
C++学习笔记(十)——向上造型
查看>>
2017/6/16
查看>>
LeetCode 445——两数相加 II
查看>>
预备作业03 20162308马平川
查看>>
【Java】嵌套For循环性能优化案例
查看>>
面试了一个开发人员
查看>>
软件工程及软件项目开发流程
查看>>
关于android4.3 bluetooth4.0的那些事儿
查看>>
嵌入式成长轨迹14 【嵌入式环境及基础】【Linux下的C编程 上】【gcc、gdb和GNU Make】...
查看>>
C语言讲义——变量的输出
查看>>
shell脚本 ----每天学一点shell
查看>>
FZU2150 :Fire Game (双起点BFS)
查看>>
php_常用操作_读取文件_数据库操作
查看>>
Linux中GCC源码编译安装
查看>>
equals与==关于Object覆盖和重载问题
查看>>
KVO
查看>>
js基础教程四之无缝滚动
查看>>
关于C51 keil使用中.c文件的链接心得
查看>>