再次用runtime的一次实践

好久不用,再次使用runtime重写代码。就用高性能添加图片圆角来再一次实践一下runtime的基本用法。runtime使用场景:- category添加关联属性- MethodSwizzle替换/交换系统方法平常使用cornerRadius和maskToBounds组合设置圆角

category添加关联属性

废话不多说,直接上代码创建一个`NSError`的category,添加一个`errorMsg`的属性。因为category本身不能添加属性,这里是使用runtime动态添加关联属性![NSError+Msg_h](/img/NSError+Msg_h.png)![NSError+Msg_m](/img/NSError+Msg_m.png)

MethodSwizzle交换系统方法

创建一个`UIImageView`的category,在分类中交换(exchange,注意这里不是替换replace)系统的`setImage:` 方法.我们要交换一个类的系统方法,那么首先想想在哪个方法中交换最合适呢?我们知道app在启动之后,会进行类注册,调用类的`+load()`方法,且只调用一次。(注意与`+initialize`的区别)[第51条:load与initialize的区别](http://polomana.com/2016/03/23/Effective-Objective-C-2-0-要点/)所以我们在分类中重写`load`方法中添加交换方法。首先我们借用AFNetworking中AFURLSessionManager中定义的swizzle的内联函数

{% codeblock lang:objc %}static inline void hl_swizzleSelector(Class theClass, SEL originalSelector, SEL swizzledSelector) {    Method originalMethod = class_getInstanceMethod(theClass, originalSelector);    Method swizzledMethod = class_getInstanceMethod(theClass, swizzledSelector);    method_exchangeImplementations(originalMethod, swizzledMethod);}static inline BOOL hl_addMethod(Class theClass, SEL selector, Method method) {    /*    YES if the method was added successfully, otherwise NO (for example, the class already contains a method implementation with that name)    */    return class_addMethod(theClass, selector,  method_getImplementation(method),  method_getTypeEncoding(method));}{% endcodeblock %}然后在`+load`方法中添加如下代码:

```objc  hl_swizzleSelector([self class], @selector(setImage:), @selector(hl_setImage:));```

/* 这里参考的是AFURLSessionManager中的方式来添加方法,但是实际上在这个情况下是永远返回NO的 * 原因:因为我们是在UIImageView的category中添加了hl_setImage:,所以在UIImageView中已经存在了该方法,再调用class_addMethod就会返回NO。 * 因此下面这些代码如果写在category中是不需要的了。 */```objc Method hlSetImageMethod = class_getInstanceMethod([self class],@selector(hl_setImage:)); BOOL result = hl_addMethod([self class],@selector(hl_setImage:), hlSetImageMethod); NSLog(@"%d",result); if (result) { }```测试一下代码:```objcUIImageView * v = [[UIImageView alloc]initWithFrame:CGRectMake(0, 0, 240, 240)];v.center = self.view.center;[self.view addSubview:v];UIImage * image = [UIImage imageNamed:@"IMG_0730.jpg"];//会产生混合图层v.layer.cornerRadius = v.frame.size.width / 2;v.layer.masksToBounds = YES;//采用UIGraphicImageContext重绘后,解决混合图层问题[v setImage:[image hl_imageByCroppingForSize:CGSizeMake(240, 240) fillColor:[UIColor whiteColor]]];//使用Method swizzle交换setImage和hl_setImage:之后:v.image = image;```

其中的性能问题

我们在UIImage+HLAdd分类中使用UIGraphicImageContext来重新绘图,这肯定是要耗时的操作。```objcCFTimeInterval start = CACurrentMediaTime();...//绘图...NSLog(@"%f",CACurrentMediaTime() - start);``````c2016-11-22 17:53:42.614 iOS-Kick-On[47630:2598373] 0.003525```大概耗时在0.003~0.006之间,以纳秒进行运算的cpu来说,还是一个稍微耗时的操作,因为我们可以把它放在后台线程来做,然后采用回调来获取返回的image```objcdispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{    UIImage * result = [self hl_imageByCroppingCorner:radius forSize:targetSize fillColor:[UIColor whiteColor]];    dispatch_async(dispatch_get_main_queue(), ^{        if (completion) {            completion(result);        }    });});```[本篇文章代码](https://github.com/koalahl/UIImageViewHack)

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • 转至元数据结尾创建: 董潇伟,最新修改于: 十二月 23, 2016 转至元数据起始第一章:isa和Class一....
    40c0490e5268阅读 1,768评论 0 9
  • 本文转载自:http://yulingtianxia.com/blog/2014/11/05/objective-...
    ant_flex阅读 784评论 0 1
  • runtime 和 runloop 作为一个程序员进阶是必须的,也是非常重要的, 在面试过程中是经常会被问到的, ...
    made_China阅读 1,226评论 0 7
  • 我们常常会听说 Objective-C 是一门动态语言,那么这个「动态」表现在哪呢?我想最主要的表现就是 Obje...
    Ethan_Struggle阅读 2,231评论 0 7
  • 文中的实验代码我放在了这个项目中。 以下内容是我通过整理[这篇博客] (http://yulingtianxia....
    茗涙阅读 946评论 0 6