JavaScriptCore实现JS调用原生方法的几种方式对比

WKWebView

如果是iOS 8及以上使用WKWebView加载网页。通过WKWebViewConfiguration添加JavaScript Message Handler的方式实现JavaScript调用原生方法。JavaScript可以给原生方法传递NSNumber, NSString, NSDate, NSArray,NSDictionary, NSNull类型的值。

比如:

WKWebViewConfiguration *config = [[WKWebViewConfiguration alloc]init];
[config.userContentController addScriptMessageHandler:self name:@"iOSInjection"];
myWKWebView = [[WKWebView alloc]initWithFrame:rect configuration:config];

接下来,在JavaScript的方法中调用window.webkit.messageHandlers.iOSInjection.postMessage([var1, var2]);方法,比如:

function wk_simpleCall() {
     window.webkit.messageHandlers.iOSInjection.postMessage("No params method called");
}
function wk_complexCall(var1, var2) {
     window.webkit.messageHandlers.iOSInjection.postMessage([var1, var2]);
}

注意这里的iOSInjection需要和前端约定好。

JavaScript抛出消息后,再接下来在原生代码的回调方法里就可以接收到消息,可以在回调方法中根据消息类型判断作何操作。

#pragma mark---WKScriptMessageHandler
//收到js注入回调
- (void)userContentController:(WKUserContentController *)userContentController didReceiveScriptMessage:(WKScriptMessage *)message
{
    //此处的message.name为iOSInjection message.body中体现js传过来的值
    NSLog(@"收到js注入  message.ame=%@  message.body=%@",message.name,message.body);
}

不能通过Block的方式实现JavaScript调用原生方法,因为拿不到WKWebView的JS上下文。

UIWebView

如果是iOS 7及以下使用UIWebView加载网页。有两种方式实现JavaScript调用原生方法。一种是Block方式,一种是利用JSExport,但都必须在webViewDidFinishLoad:回调方法中设置相关Block和JSExport才行。

  • Block方式实现JavaScript调用原生方法

webViewDidFinishLoad:回调方法中通过拿到JavaScript上下文,然后注入相关Block即可,相当于在JavaScript上下文环境中新增一个对象,那么JavaScript端就可以直接使用这个对象了。

- (void)webViewDidFinishLoad:(UIWebView *)webView
{
    //UIWebView需要使用这种方式获取JS上下文,WKWebView无法拿到!
    JSContext *context = [webView valueForKeyPath:@"documentView.webView.mainFrame.javaScriptContext"];
    //这里需要避免循环引用
    //创建函数
    context[@"callOcMethod"] = ^(){
        NSArray *args = [JSContext currentArguments];
        NSLog(@"args=%@", args);
    };
}

上面的代码执行完,JavaScript端就可以像这样用callOcMethod这个对象了:

function ui_simpleCall() {
     callOcMethod();
}
function ui_complexCall(var1, var2) {
     callOcMethod({"var1": var1, "var2": var2});
}
  • JSExport方式实现JavaScript调用原生方法

JSExport:用来将原生方法输出给JavaScript。其本身并没有任何属性或方法,需要自己定义一个遵守JSExport的协议,然后在协议中声明方法,然后在webViewDidFinishLoad:回调方法中往JavaScript环境中输出一个遵守该协议的对象,最后JavaScript就可以通过这个对象来调用协议中声明的方法了。

第一步:定义协议

//需要自己定义一个遵守JSExport协议的协议
@protocol YLTestExport <JSExport>

//方式1、定义方法
-(void)log:(NSString *)string type:(NSInteger)type;


//方式2、JS调用logName('test', 123) 时,将自动调用logName:withType:回调方法
JSExportAs(logName, - (void)logName:(NSString *)name withType:(NSInteger)type);

@end

第二步:把一个遵守该协议的对象输出给JavaScript

这里我们将当前UIViewController设为遵守YLTestExport协议,并实现协议中定义的方法。

- (void)webViewDidFinishLoad:(UIWebView *)webView
{
    //方式2、JSExport方式
    JSContext *context = [webView valueForKeyPath:@"documentView.webView.mainFrame.javaScriptContext"];
    context[@"currentController"] = self;
}

第三步:JavaScript通过该对象调用协议中定义的方法

function ui_simpleExportCall() {
<!--这里的currentController和logName方法要现在原生代码里面定义好-->
     currentController.logName('Hello world!', 1024);//注意这里是logName方法,并不是logNameWithType方法
     //或者
     currentController.logType('Hello world!', 1024)
}

注意:在YLTestExport协议中声明方法时有两种方式:

一种是直接定义,像-(void)log:(NSString *)string type:(NSInteger)type这样,调用的时候冒号后面的首字母需要转为大写,比如这里就是logType

另一种是通过JSExportAs这个宏定义来声明,第一个参数相当于给对应的方法取一个简称,这样在JavaScript调用该方法时就可以直接使用这个简称了。

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

推荐阅读更多精彩内容

  • 随着H5技术的兴起,在iOS开发过程中,难免会遇到原生应用需要和H5页面交互的问题。其中会涉及方法调用及参数传值等...
    Chris_js阅读 3,050评论 1 8
  • 虽然WKWebView是在Apple的WWDC 2014随iOS 8和OS X 10.10出来的,是为了解决UIW...
    winann阅读 135,093评论 196 641
  • 最近整理了一下原生与H5之间的交互方式,简单的做个总结。OC端与JS的交互,大致有这几种:拦截协议、JavaScr...
    谈Xx阅读 31,083评论 41 75
  • 跟原生开发相比,H5的开发相对来一个成熟的框架和团队来讲在开发速度和开发效率上有着比原生很大的优势,至少不用等待审...
    大冲哥阅读 1,823评论 0 7
  • 人生的太多第一次,都是自己一个人。一个人来到这世界,此时的我又要一个人去尝试融入这社会,要去创造自己的未来。终于我...
    东方优玛阅读 142评论 0 0