iOS开发免不了要与UIWebView打交道,然后就要涉及到JS与原生OC交互,今天总结一下JS与原生OC交互的两种方式。

JS调用原生OC篇

方式一

第一种方式是用JS发起一个假的URL请求,然后利用UIWebView的代理方法拦截这次请求,然后再做相应的处理。
我写了一个简单的HTML网页和一个btn点击事件用来与原生OC交互,HTML代码如下:

<html><header><meta http-equiv="Content-Type" content="text/html; charset=utf-8" />
<script type="text/javascript">function showAlert(message){alert(message);}function loadURL(url){var iFrame;iFrame = document.createElement("iframe");iFrame.setAttribute("src", url);iFrame.setAttribute("style", "display:none;");iFrame.setAttribute("height", "0px");iFrame.setAttribute("width", "0px");iFrame.setAttribute("frameborder", "0");document.body.appendChild(iFrame);// 发起请求后这个 iFrame 就没用了,所以把它从 dom 上移除掉iFrame.parentNode.removeChild(iFrame);iFrame = null;}function firstClick(){loadURL("firstClick://shareClick?title=分享的标题&content=分享的内容&url=链接地址&imagePath=图片地址");}</script></header><body><h2> 这里是第一种方式 </h2><br/><br/><button type="button" onclick="firstClick()">Click Me!</button></body>
</html>

然后在项目的控制器中实现UIWebView的代理方法:

#pragma mark - UIWebViewDelegate
- (BOOL)webView:(UIWebView *)webView shouldStartLoadWithRequest:(NSURLRequest *)request navigationType:(UIWebViewNavigationType)navigationType
{NSURL * url = [request URL];if ([[url scheme] isEqualToString:@"firstclick"]) {NSArray *params =[url.query componentsSeparatedByString:@"&"];NSMutableDictionary *tempDic = [NSMutableDictionary dictionary];for (NSString *paramStr in params) {NSArray *dicArray = [paramStr componentsSeparatedByString:@"="];if (dicArray.count > 1) {NSString *decodeValue = [dicArray[1] stringByReplacingPercentEscapesUsingEncoding:NSUTF8StringEncoding];[tempDic setObject:decodeValue forKey:dicArray[0]];}}UIAlertView *alertView = [[UIAlertView alloc] initWithTitle:@"方式一" message:@"这是OC原生的弹出窗" delegate:self cancelButtonTitle:@"收到" otherButtonTitles:nil];[alertView show];NSLog(@"tempDic:%@",tempDic);return NO;}return YES;
}

注意:1. JS中的firstClick,在拦截到的url scheme全都被转化为小写。
2.html中需要设置编码,否则中文参数可能会出现编码问题。
3.JS用打开一个iFrame的方式替代直接用document.location的方式,以避免多次请求,被替换覆盖的问题。

早期的JS与原生交互的开源库很多都是用得这种方式来实现的,例如:PhoneGap、WebViewJavascriptBridge。关于这种方式调用OC方法,唐巧早期有篇文章有过介绍:
关于UIWebView和PhoneGap的总结

方式二

在iOS 7之后,apple添加了一个新的库JavaScriptCore,用来做JS交互,因此JS与原生OC交互也变得简单了许多。
首先导入JavaScriptCore库, 然后在OC中获取JS的上下文

JSContext*context = [self.webView valueForKeyPath:@"documentView.webView.mainFrame.javaScriptContext"];

再然后定义好JS需要调用的方法,例如JS要调用share方法:
则可以在UIWebView加载url完成后,在其代理方法中添加要调用的share方法:

- (void)webViewDidFinishLoad:(UIWebView *)webView
{JSContext *context = [self.webView valueForKeyPath:@"documentView.webView.mainFrame.javaScriptContext"];//定义好JS要调用的方法, share就是调用的share方法名context[@"share"] = ^() {NSLog(@"+++++++Begin Log+++++++");NSArray *args = [JSContext currentArguments];dispatch_async(dispatch_get_main_queue(), ^{UIAlertView *alertView = [[UIAlertView alloc] initWithTitle:@"方式二" message:@"这是OC原生的弹出窗" delegate:self cancelButtonTitle:@"收到" otherButtonTitles:nil];[alertView show];});for (JSValue *jsVal in args) {NSLog(@"%@", jsVal.toString);}NSLog(@"-------End Log-------");};
}

注意:
可能最新版本的iOS系统做了改动,现在(iOS9,Xcode 7.3,去年使用Xcode 6 和iOS 8没有线程问题)中测试,block中是在子线程,因此执行UI操作,控制台有警告,需要回到主线程再操作UI。

其中相对应的html部分如下:

<html><header><meta http-equiv="Content-Type" content="text/html; charset=utf-8" /><script type="text/javascript">function secondClick(){share('分享的标题','分享的内容','图片地址');}function showAlert(message){alert(message);}</script></header><body><h2> 这里是第二种方式 </h2><br/><br/><button type="button" onclick="secondClick()">Click Me!</button></body>
</html>

JS部分确实要简单的多了。

OC调用JS篇

方式一

NSString *jsStr = [NSString stringWithFormat:@"showAlert('%@')",@"这里是JS中alert弹出的message"];
[_webView stringByEvaluatingJavaScriptFromString:jsStr];

注意:该方法会同步返回一个字符串,因此是一个同步方法,可能会阻塞UI。

方式二

继续使用JavaScriptCore库来做JS交互。

JSContext*context = [self.webView valueForKeyPath:@"documentView.webView.mainFrame.javaScriptContext"];
NSString *textJS = @"showAlert('这里是JS中alert弹出的message')";
[context evaluateScript:textJS];

重点:
stringByEvaluatingJavaScriptFromString是一个同步的方法,使用它执行JS方法时,如果JS 方法比较耗的时候,会造成界面卡顿。尤其是js 弹出alert 的时候。
alert 也会阻塞界面,等待用户响应,而stringByEvaluatingJavaScriptFromString又会等待js执行完毕返回。这就造成了死锁。
官方推荐使用WKWebViewevaluateJavaScript:completionHandler:代替这个方法。
其实我们也有另外一种方式,自定义一个延迟执行alert 的方法来防止阻塞,然后我们调用自定义的alert 方法。同理,耗时较长的js 方法也可以放到setTimeout 中。

function asyncAlert(content){setTimeout(function(){alert(content);},1);
}

iOS下JS与原生OC互相调用相关推荐

  1. iOS下JS和原生交互,函数互调

    现在越来越多的APP都是H5和原生混合开发,这样确实方便快捷,但是H5的部分总避免不了很多与原生的交互,原生调JS函数还比较简单,原生的API函数stringByEvaluatingJavaScrip ...

  2. iOS下JS与OC互相调用(一)--UIWebView 拦截URL

    1.在JS 中做一次URL跳转,然后在OC中拦截跳转.(这里分为UIWebView 和 WKWebView两种,去年因为还要兼容iOS 6,所以没办法只能采用UIWebView来做.) 2.利用WKW ...

  3. iOS下JS与OC互相调用(五)--UIWebView + WebViewJavascriptBridge

    WebViewJavascriptBridge是一个有点年代的JS与OC交互的库,使用该库的著名应用还挺多的,目前这个库有7000+star.我去翻看了它的第一版本已经是4年前了,在版本V4.1.4以 ...

  4. iOS下JS与OC互相调用(六)--WKWebView + WebViewJavascriptBridge

    2019独角兽企业重金招聘Python工程师标准>>> iOS下JS与OC互相调用(六)--WKWebView + WebViewJavascriptBridge 转载:原地址 ht ...

  5. iOS下JS与OC互相调用(四)--JavaScriptCore

    前面讲完拦截URL的方式实现JS与OC互相调用,终于到JavaScriptCore了.它是从iOS7开始加入的,用 Objective-C 把 WebKit 的 JavaScript 引擎封装了一下, ...

  6. iOS下JS与OC互相调用(三)--MessageHandler

    使用WKWebView的时候,如果想要实现JS调用OC方法,除了拦截URL之外,还有一种简单的方式.那就是利用WKWebView的新特性MessageHandler来实现JS调用原生方法. Messa ...

  7. iOS下JS与OC互相调用(二)--WKWebView 拦截URL

    在上篇文章中讲述了使用UIWebView拦截URL的方式来处理JS与OC交互. 由于UIWebView比较耗内存,性能上不太好,而苹果在iOS 8中推出了WKWebView. 同样的用WKWebVie ...

  8. ios下js复制到粘贴板_h5实现一键复制到粘贴板 兼容ios

    实现原理 采用document.execCommand('copy') 来实现复制到粘贴板功能 复制必须是选中input框的文字内容,然后执行document.execCommand('copy') ...

  9. iOS下OpenCV开发用OC还是Swift

    本文为作者原创,转载请注明出处(http://www.cnblogs.com/mar-q/)by 负赑屃 其实标题中这个问题并不准确,准确的说法应该是iOS下的OpenCV开发是使用OC还是Swift ...

最新文章

  1. Failed to create the Java Virtual Machine
  2. Kick Start 2019 Round D
  3. NET问答: 如何使用 C# 自动化关闭电脑 ?
  4. JavaScript 清除图片背景颜色 使之透明
  5. 分别统计出其中英文字母、空格、数字和其它字符的个数 matlab 程序,编写一段程序,要求先输入一行字符,然后分别统计出其中英文...
  6. mysql存储过程返回多个值_数据库mysql存储过程之返回多个值的方法示例
  7. Project Euler 97 :Large non-Mersenne prime 非梅森大素数
  8. svn 备份恢复【转】
  9. const,readonly字段的取舍!
  10. 【ArcGIS|空间分析】利用DEM提取山脊、山谷线(水文分析、表面分析)
  11. 直流稳压电源设计(单相)_电力电子课程设计
  12. matlab资产组合最优配置,[转载]资产组合有效前沿的解和最优解(MATLAB语言)
  13. php小卖铺源码,PHP自动化售货发卡网源码-小酒资源
  14. 【PR #2】史莱姆(值域分段)
  15. 微信扫码支付测试用例设计
  16. 支付宝二维码支付文档,找了好久,记在这里
  17. 软件工程可行性研究报告
  18. 青岛地铁不能让你的房子升值
  19. java中的math.abs_java – Math.abs(a – b)的更快实现 – Math.abs(c – d)?
  20. Ubuntu下连上网络开始学习啦

热门文章

  1. redis源码分析--zslRandomLevel位运算解析
  2. 【Qt】在ubuntu14.04上安装Qt5.12(失败)
  3. cpu渲染测试软件,CineBench渲染能力测试_CPUCPU评测-中关村在线
  4. 【java】兴唐课程第五节到第九节知识点总结
  5. spark编程基础--5.3数据读写
  6. 慕课《深入理解计算机系统》袁林枫老师章节测试1-9
  7. android专栏目录
  8. 小程序地图多个 circles 使用demo
  9. linux基础—课堂随笔010_系统启动和内核管理
  10. 083、Prometheus架构(2019-05-05 周日)