iOS Method Swizzling 的一个实际应用

本文 Method Swizzling 部分参考了 《iOS黑魔法-Method Swizzling》

一、问题

最近在维护公司一个久远的项目时,发现当时使用了 UIWebView 展示 HTML 页面,为了解决 JavaScript 中 alert 和 confirm 样式不能自定义的问题,所以通过实现以下方法,

@interface UIWebView (JSConfirmAlert)

- (void)webView:(UIWebView *)sender runJavaScriptAlertPanelWithMessage:(NSString *)message initiatedByFrame:(id)frame;
- (BOOL)webView:(UIWebView *)sender runJavaScriptConfirmPanelWithMessage:(NSString *)message initiatedByFrame:(id)frame;

@end

来拦截 alert 和 confirm ,并通过 UIAlertView 重新实现。

但是原来的代码在实现上有些问题,这部分功能又是做成了 framework 集成到工程里面的,而且由于项目几经易手,framework 部分的源码已经没有了。由于原来是通过给 UIWebView 写 Category 来实现的功能,所以即便不引用 .h 文件,只要集成了 framework,这部分代码就会生效。(这一点是刚发现的,原来一直以为要引用了 .h 才会生效。)

二、解决方案

无奈之下,就考虑用 Method Swizzling 来改写之前的方法了。

// UIWebView+SwizzlingAlertAndConfirm.h

#import <Foundation/Foundation.h>

@interface UIWebView (SwizzlingAlertAndConfirm)

@end

// UIWebView+SwizzlingAlertAndConfirm.m

#import <UIKit/UIKit.h>
#import <objc/runtime.h>
#import "UIWebView+SwizzlingAlertAndConfirm.h"

@implementation UIWebView (SwizzlingAlertAndConfirm)

+ (void)exchangeMethod:(SEL)fromSelector toMethod:(SEL)toSelector {

    // 通过class_getInstanceMethod()函数从当前对象中的method list获取method结构体,如果是类方法就使用class_getClassMethod()函数获取。
    Method fromMethod = class_getInstanceMethod([self class], fromSelector);
    Method toMethod = class_getInstanceMethod([self class], toSelector);

    /**
     *  我们在这里使用class_addMethod()函数对Method Swizzling做了一层验证,如果self没有实现被交换的方法,会导致失败。
     *  而且self没有交换的方法实现,但是父类有这个方法,这样就会调用父类的方法,结果就不是我们想要的结果了。
     *  所以我们在这里通过class_addMethod()的验证,如果self实现了这个方法,class_addMethod()函数将会返回NO,我们就可以对其进行交换了。
     */
    if (!class_addMethod([self class], fromSelector, method_getImplementation(toMethod), method_getTypeEncoding(toMethod))) {
        method_exchangeImplementations(fromMethod, toMethod);
    }

}


+ (void)load {
    [super load];

    [self exchangeMethod:@selector(webView:runJavaScriptAlertPanelWithMessage:initiatedByFrame:)
                toMethod:@selector(swizzlingWebView:runJavaScriptAlertPanelWithMessage:initiatedByFrame:)];

    [self exchangeMethod:@selector(webView:runJavaScriptConfirmPanelWithMessage:initiatedByFrame:)
                toMethod:@selector(swizzlingWebView:runJavaScriptConfirmPanelWithMessage:initiatedByFrame:)];

    [self exchangeMethod:@selector(alertView:clickedButtonAtIndex:)
                toMethod:@selector(swizzlingAlertView:clickedButtonAtIndex:)];

}



- (void)swizzlingWebView:(UIWebView *)sender runJavaScriptAlertPanelWithMessage:(NSString *)message initiatedByFrame:(id)frame {
    UIAlertView* customAlert = [[UIAlertView alloc] initWithTitle:@"助手提示" message:message delegate:nil cancelButtonTitle:@"确定bbb" otherButtonTitles:nil];
    [customAlert show];
}

static BOOL diagStat = NO;
static NSInteger bIdx = -1;

- (BOOL)swizzlingWebView:(UIWebView *)sender runJavaScriptConfirmPanelWithMessage:(NSString *)message initiatedByFrame:(id)frame {
    UIAlertView *confirmDiag = [[UIAlertView alloc] initWithTitle:@"助手提示"
                                                          message:message
                                                         delegate:self
                                                cancelButtonTitle:@"取消13"
                                                otherButtonTitles:@"确定13", nil];

    [confirmDiag show];
    bIdx = -1;

    while (bIdx==-1) {
        //[NSThread sleepForTimeInterval:0.2];
        [[NSRunLoop mainRunLoop] runUntilDate:[NSDate dateWithTimeIntervalSinceNow:0.1f]];
    }
    if (bIdx == 0){//取消;
        diagStat = NO;
    }
    else if (bIdx == 1) {//确定;
        diagStat = YES;
    }
    return diagStat;
}

- (void)swizzlingAlertView:(UIAlertView *)alertView clickedButtonAtIndex:(NSInteger)buttonIndex{
    bIdx = buttonIndex;
}


@end

工程引用这两个文件后,问题解决。

三、讨论
1、Method Swizzling 的封装

这里对 Method Swizzling 做了个简单的封装,不过只是为了写着方便随便整了下。真正在项目中我们肯定会在很多地方用到 Method Swizzling,而且在使用这个特性时有很多需要注意的地方。我们可以将 Method Swizzling 封装起来,也可以使用一些比较成熟的第三方。在这里我推荐Github上星最多的一个第三方——JRSwizzle

2、改进

都折腾完之后才发现,就这个项目本身的问题而言,其实用不到 Method Swizzling,只要写个 UIWebView 的子类,在子类中重新实现这几个方法,然后直接使用子类就好了。当时没想起来。不过使用子类化的方法,还是需要修改原来代码中对 UIWebView 引用的那部分代码。而使用 Method Swizzling 还是更酷一些,原来的代码不用做任何修改,只要在工程中引入这两个文件就可以了。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 160,108评论 4 364
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 67,699评论 1 296
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 109,812评论 0 244
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 44,236评论 0 213
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 52,583评论 3 288
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 40,739评论 1 222
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 31,957评论 2 315
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 30,704评论 0 204
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 34,447评论 1 246
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 30,643评论 2 249
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 32,133评论 1 261
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 28,486评论 3 256
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 33,151评论 3 238
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 26,108评论 0 8
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 26,889评论 0 197
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 35,782评论 2 277
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 35,681评论 2 272

推荐阅读更多精彩内容

  • 有些钱真不能省。 之前为了省钱,买了个半自动的洗衣机。真是坑死我了:每次接水要自己控制水量。洗完后,排水要自己放。...
    alabiubiubiu阅读 192评论 0 2
  • 今天的输出是信息安全知识三的导图。
    huang_yw阅读 154评论 0 0
  • 家人们,晚上好! 真心体会到,有些事早做比晚做要好。就比如心里想的写信的事。哈哈。一会儿要去打球,回家估计又要晚了...
    赵诚彬阅读 264评论 0 0