用JavaScriptCore做android和iOS都兼容的JS-NativeSDK(****)
2015-12-09 15:55
435 查看
转载自:http://blog.csdn.net/u010124617/article/details/50053069
最近在给公司做一个JS-Native的SDK,就是用于JS和原生之间的交互。
使用场景上主要还是webView,那么原先的url拦截的方式已经不再考虑,我们使用了iOS7之后的JavaScriptCore.framework。
总体来说,苹果的JavaScriptCore的API是非常易用的,主要操作步骤如下:
1.获取一个JSContext。
2.处理某个JS端调起的方法,比如log
3.调用JS端执行某个JS方法
4.重定义某个JS端的方法(其实和3是一种,不过特殊了点)
掌握了以上4步,基本上在iOS端和JS的交互就完全OK了。
接着我们开始和android端h5端进行联调。
到这里,我们发现了一个问题:
android端JS方法调用原生方法时,需要指定interfaceName,所以他们的JS调用方式是这样的:
前面多出来了xxname,这种方式下,我们直接使用JSContext进行监听处理是不行的
那么怎么办呢?胜利在望的时候卡住了?难道要让h5那边调用的时候按平台区分么?如果是iOS平台就调用log(‘xx’),android就调用xxname.log(‘xx’)??
还好,我们这么犀利怎么会解决不了这样的问题。
这里我给大家介绍两种方法解决:
方案一:
使用JSExport,达到和android类似处理效果
步骤如下:
1.定义WebViewJSHelperProtocol协议实线JSExport协议
并在协议中定义出所有JS端要调用的方法
2.定义类实现该协议
3.注册该类的实例到JSContext
到此,JS调用xxname.log方法时,会进入我们iOS端定义的log方法中执行。
关于JSExport还有一些定义别名之类的,感兴趣的同学可以自行看API或网上文档。
方案二:
这个方案的操作方式就简单多了,对比方案一,无需定义实现的原生方法,操作如下:
1.随便定义一个类加上JSExport协议
2.注册给JSContext
3.处理某个JS端调起的方法
好了,现在整套功能都可以使用了。
上面方案二的方式个人觉得优点更多一些,首先更贴近JS习惯,其次不需要定义那么多的方法,而且对于js端调用来说,不定参数个数也更容易些。
这篇文章的最后,讲一点小的注意事项:
1. 使用 NSArray *args = [JSContext currentArguments];得到的args,内部都是JSValue,千万注意这一点,如果想要使用,最好转成相应的类型,比如[obj toString]之类的,具体可以看JSValue的头文件
2. 要注意循环引用的问题,不要写出 _jsContext[@”xxname”] = self; 这样的代码来,其他的就是block中的了,老生常谈不再细说。
3. 要注意JS端和原生端生命周期的问题
两端的内存管理方式不一样,JS端是GC,我们是引用计数,一般情况不会有生命周期的问题,但是必要时,需要使用JSVirtualMachine包装一下,这块我也还在学习,希望大家注意。
最近在给公司做一个JS-Native的SDK,就是用于JS和原生之间的交互。
使用场景上主要还是webView,那么原先的url拦截的方式已经不再考虑,我们使用了iOS7之后的JavaScriptCore.framework。
总体来说,苹果的JavaScriptCore的API是非常易用的,主要操作步骤如下:
1.获取一个JSContext。
<code class="hljs bash has-numbering" style="display: block; padding: 0px; background-color: transparent; color: inherit; box-sizing: border-box; font-family: 'Source Code Pro', monospace;font-size:undefined; white-space: pre; border-top-left-radius: 0px; border-top-right-radius: 0px; border-bottom-right-radius: 0px; border-bottom-left-radius: 0px; word-wrap: normal; background-position: initial initial; background-repeat: initial initial;">_jsContext = [_webView valueForKeyPath: @<span class="hljs-string" style="color: rgb(0, 136, 0); box-sizing: border-box;">"documentView.webView.mainFrame.javaScriptContext"</span>];</code><ul class="pre-numbering" style="box-sizing: border-box; position: absolute; width: 50px; background-color: rgb(238, 238, 238); top: 0px; left: 0px; margin: 0px; padding: 6px 0px 40px; border-right-width: 1px; border-right-style: solid; border-right-color: rgb(221, 221, 221); list-style: none; text-align: right;"><li style="box-sizing: border-box; padding: 0px 5px;">1</li><li style="box-sizing: border-box; padding: 0px 5px;">2</li></ul>
2.处理某个JS端调起的方法,比如log
<code class="hljs objectivec has-numbering" style="display: block; padding: 0px; background-color: transparent; color: inherit; box-sizing: border-box; font-family: 'Source Code Pro', monospace;font-size:undefined; white-space: pre; border-top-left-radius: 0px; border-top-right-radius: 0px; border-bottom-right-radius: 0px; border-bottom-left-radius: 0px; word-wrap: normal; background-position: initial initial; background-repeat: initial initial;">_jsContext[@<span class="hljs-string" style="color: rgb(0, 136, 0); box-sizing: border-box;">"log"</span>] = ^() { <span class="hljs-built_in" style="color: rgb(102, 0, 102); box-sizing: border-box;">NSArray</span> *args = [JSContext currentArguments]; <span class="hljs-keyword" style="color: rgb(0, 0, 136); box-sizing: border-box;">for</span> (<span class="hljs-keyword" style="color: rgb(0, 0, 136); box-sizing: border-box;">id</span> obj in args) { <span class="hljs-comment" style="color: rgb(136, 0, 0); box-sizing: border-box;">//需要注意这里的obj都还是JSValue</span> <span class="hljs-built_in" style="color: rgb(102, 0, 102); box-sizing: border-box;">NSLog</span>(@<span class="hljs-string" style="color: rgb(0, 136, 0); box-sizing: border-box;">"%@"</span>,obj); } };</code><ul class="pre-numbering" style="box-sizing: border-box; position: absolute; width: 50px; background-color: rgb(238, 238, 238); top: 0px; left: 0px; margin: 0px; padding: 6px 0px 40px; border-right-width: 1px; border-right-style: solid; border-right-color: rgb(221, 221, 221); list-style: none; text-align: right;"><li style="box-sizing: border-box; padding: 0px 5px;">1</li><li style="box-sizing: border-box; padding: 0px 5px;">2</li><li style="box-sizing: border-box; padding: 0px 5px;">3</li><li style="box-sizing: border-box; padding: 0px 5px;">4</li><li style="box-sizing: border-box; padding: 0px 5px;">5</li><li style="box-sizing: border-box; padding: 0px 5px;">6</li><li style="box-sizing: border-box; padding: 0px 5px;">7</li></ul>
3.调用JS端执行某个JS方法
<code class="hljs ini has-numbering" style="display: block; padding: 0px; background-color: transparent; color: inherit; box-sizing: border-box; font-family: 'Source Code Pro', monospace;font-size:undefined; white-space: pre; border-top-left-radius: 0px; border-top-right-radius: 0px; border-bottom-right-radius: 0px; border-bottom-left-radius: 0px; word-wrap: normal; background-position: initial initial; background-repeat: initial initial;"><span class="hljs-title" style="box-sizing: border-box; color: rgb(0, 102, 102);">[_jsContext evaluateScript:@"log('arg1')"]</span><span class="hljs-comment" style="color: rgb(136, 0, 0); box-sizing: border-box;">;</span> <span class="hljs-title" style="box-sizing: border-box; color: rgb(0, 102, 102);">[_jsContext evaluateScript:@"logCallback('arg1')"]</span><span class="hljs-comment" style="color: rgb(136, 0, 0); box-sizing: border-box;">;</span></code><ul class="pre-numbering" style="box-sizing: border-box; position: absolute; width: 50px; background-color: rgb(238, 238, 238); top: 0px; left: 0px; margin: 0px; padding: 6px 0px 40px; border-right-width: 1px; border-right-style: solid; border-right-color: rgb(221, 221, 221); list-style: none; text-align: right;"><li style="box-sizing: border-box; padding: 0px 5px;">1</li><li style="box-sizing: border-box; padding: 0px 5px;">2</li><li style="box-sizing: border-box; padding: 0px 5px;">3</li><li style="box-sizing: border-box; padding: 0px 5px;">4</li></ul>
4.重定义某个JS端的方法(其实和3是一种,不过特殊了点)
<code class="hljs tex has-numbering" style="display: block; padding: 0px; background-color: transparent; color: inherit; box-sizing: border-box; font-family: 'Source Code Pro', monospace;font-size:undefined; white-space: pre; border-top-left-radius: 0px; border-top-right-radius: 0px; border-bottom-right-radius: 0px; border-bottom-left-radius: 0px; word-wrap: normal; background-position: initial initial; background-repeat: initial initial;"><span class="hljs-special" style="box-sizing: border-box; color: rgb(102, 102, 0);">[</span>_jsContext evaluateScript:@"checkAPI = function()<span class="hljs-special" style="box-sizing: border-box; color: rgb(102, 102, 0);">{</span><span class="hljs-command" style="box-sizing: border-box; color: rgb(0, 0, 136);">\ </span> return <span class="hljs-special" style="box-sizing: border-box; color: rgb(102, 102, 0);">[</span><span class="hljs-command" style="box-sizing: border-box; color: rgb(0, 0, 136);">\ </span> 'selectImage',<span class="hljs-command" style="box-sizing: border-box; color: rgb(0, 0, 136);">\ </span> 'startRecord',<span class="hljs-command" style="box-sizing: border-box; color: rgb(0, 0, 136);">\ </span> 'login',<span class="hljs-command" style="box-sizing: border-box; color: rgb(0, 0, 136);">\ </span> <span class="hljs-special" style="box-sizing: border-box; color: rgb(102, 102, 0);">]</span>;<span class="hljs-command" style="box-sizing: border-box; color: rgb(0, 0, 136);">\ </span> <span class="hljs-special" style="box-sizing: border-box; color: rgb(102, 102, 0);">}</span>"<span class="hljs-special" style="box-sizing: border-box; color: rgb(102, 102, 0);">]</span>;</code><ul class="pre-numbering" style="box-sizing: border-box; position: absolute; width: 50px; background-color: rgb(238, 238, 238); top: 0px; left: 0px; margin: 0px; padding: 6px 0px 40px; border-right-width: 1px; border-right-style: solid; border-right-color: rgb(221, 221, 221); list-style: none; text-align: right;"><li style="box-sizing: border-box; padding: 0px 5px;">1</li><li style="box-sizing: border-box; padding: 0px 5px;">2</li><li style="box-sizing: border-box; padding: 0px 5px;">3</li><li style="box-sizing: border-box; padding: 0px 5px;">4</li><li style="box-sizing: border-box; padding: 0px 5px;">5</li><li style="box-sizing: border-box; padding: 0px 5px;">6</li><li style="box-sizing: border-box; padding: 0px 5px;">7</li></ul>
掌握了以上4步,基本上在iOS端和JS的交互就完全OK了。
接着我们开始和android端h5端进行联调。
到这里,我们发现了一个问题:
android端JS方法调用原生方法时,需要指定interfaceName,所以他们的JS调用方式是这样的:
<code class="hljs rust has-numbering" style="display: block; padding: 0px; background-color: transparent; color: inherit; box-sizing: border-box; font-family: 'Source Code Pro', monospace;font-size:undefined; white-space: pre; border-top-left-radius: 0px; border-top-right-radius: 0px; border-bottom-right-radius: 0px; border-bottom-left-radius: 0px; word-wrap: normal; background-position: initial initial; background-repeat: initial initial;">xxname.<span class="hljs-keyword" style="color: rgb(0, 0, 136); box-sizing: border-box;">log</span>(<span class="hljs-string" style="color: rgb(0, 136, 0); box-sizing: border-box;">'arg1'</span>);</code><ul class="pre-numbering" style="box-sizing: border-box; position: absolute; width: 50px; background-color: rgb(238, 238, 238); top: 0px; left: 0px; margin: 0px; padding: 6px 0px 40px; border-right-width: 1px; border-right-style: solid; border-right-color: rgb(221, 221, 221); list-style: none; text-align: right;"><li style="box-sizing: border-box; padding: 0px 5px;">1</li></ul>
前面多出来了xxname,这种方式下,我们直接使用JSContext进行监听处理是不行的
<code class="hljs objectivec has-numbering" style="display: block; padding: 0px; background-color: transparent; color: inherit; box-sizing: border-box; font-family: 'Source Code Pro', monospace;font-size:undefined; white-space: pre; border-top-left-radius: 0px; border-top-right-radius: 0px; border-bottom-right-radius: 0px; border-bottom-left-radius: 0px; word-wrap: normal; background-position: initial initial; background-repeat: initial initial;"><span class="hljs-comment" style="color: rgb(136, 0, 0); box-sizing: border-box;">//这个不能被执行到</span> _jsContext[@<span class="hljs-string" style="color: rgb(0, 136, 0); box-sizing: border-box;">"xxname.log"</span>] = ^() { <span class="hljs-built_in" style="color: rgb(102, 0, 102); box-sizing: border-box;">NSArray</span> *args = [JSContext currentArguments]; <span class="hljs-keyword" style="color: rgb(0, 0, 136); box-sizing: border-box;">for</span> (<span class="hljs-keyword" style="color: rgb(0, 0, 136); box-sizing: border-box;">id</span> obj in args) { <span class="hljs-comment" style="color: rgb(136, 0, 0); box-sizing: border-box;">//需要注意这里的obj都还是JSValue</span> <span class="hljs-built_in" style="color: rgb(102, 0, 102); box-sizing: border-box;">NSLog</span>(@<span class="hljs-string" style="color: rgb(0, 136, 0); box-sizing: border-box;">"%@"</span>,obj); } };</code><ul class="pre-numbering" style="box-sizing: border-box; position: absolute; width: 50px; background-color: rgb(238, 238, 238); top: 0px; left: 0px; margin: 0px; padding: 6px 0px 40px; border-right-width: 1px; border-right-style: solid; border-right-color: rgb(221, 221, 221); list-style: none; text-align: right;"><li style="box-sizing: border-box; padding: 0px 5px;">1</li><li style="box-sizing: border-box; padding: 0px 5px;">2</li><li style="box-sizing: border-box; padding: 0px 5px;">3</li><li style="box-sizing: border-box; padding: 0px 5px;">4</li><li style="box-sizing: border-box; padding: 0px 5px;">5</li><li style="box-sizing: border-box; padding: 0px 5px;">6</li><li style="box-sizing: border-box; padding: 0px 5px;">7</li><li style="box-sizing: border-box; padding: 0px 5px;">8</li></ul>
那么怎么办呢?胜利在望的时候卡住了?难道要让h5那边调用的时候按平台区分么?如果是iOS平台就调用log(‘xx’),android就调用xxname.log(‘xx’)??
还好,我们这么犀利怎么会解决不了这样的问题。
这里我给大家介绍两种方法解决:
方案一:
使用JSExport,达到和android类似处理效果
步骤如下:
1.定义WebViewJSHelperProtocol协议实线JSExport协议
并在协议中定义出所有JS端要调用的方法
<code class="hljs objectivec has-numbering" style="display: block; padding: 0px; background-color: transparent; color: inherit; box-sizing: border-box; font-family: 'Source Code Pro', monospace;font-size:undefined; white-space: pre; border-top-left-radius: 0px; border-top-right-radius: 0px; border-bottom-right-radius: 0px; border-bottom-left-radius: 0px; word-wrap: normal; background-position: initial initial; background-repeat: initial initial;"><span class="hljs-class" style="box-sizing: border-box;"><span class="hljs-keyword" style="color: rgb(0, 0, 136); box-sizing: border-box;">@protocol</span> <span class="hljs-title" style="box-sizing: border-box; color: rgb(102, 0, 102);">WebViewJSHelperProtocol</span><<span class="hljs-title" style="box-sizing: border-box; color: rgb(102, 0, 102);">JSExport</span>></span> -(<span class="hljs-keyword" style="color: rgb(0, 0, 136); box-sizing: border-box;">void</span>)log:(<span class="hljs-built_in" style="color: rgb(102, 0, 102); box-sizing: border-box;">NSString</span>*)arg; <span class="hljs-keyword" style="color: rgb(0, 0, 136); box-sizing: border-box;">@end</span></code><ul class="pre-numbering" style="box-sizing: border-box; position: absolute; width: 50px; background-color: rgb(238, 238, 238); top: 0px; left: 0px; margin: 0px; padding: 6px 0px 40px; border-right-width: 1px; border-right-style: solid; border-right-color: rgb(221, 221, 221); list-style: none; text-align: right;"><li style="box-sizing: border-box; padding: 0px 5px;">1</li><li style="box-sizing: border-box; padding: 0px 5px;">2</li><li style="box-sizing: border-box; padding: 0px 5px;">3</li></ul>
2.定义类实现该协议
<code class="hljs objectivec has-numbering" style="display: block; padding: 0px; background-color: transparent; color: inherit; box-sizing: border-box; font-family: 'Source Code Pro', monospace;font-size:undefined; white-space: pre; border-top-left-radius: 0px; border-top-right-radius: 0px; border-bottom-right-radius: 0px; border-bottom-left-radius: 0px; word-wrap: normal; background-position: initial initial; background-repeat: initial initial;"><span class="hljs-class" style="box-sizing: border-box;"><span class="hljs-keyword" style="color: rgb(0, 0, 136); box-sizing: border-box;">@interface</span> <span class="hljs-title" style="box-sizing: border-box; color: rgb(102, 0, 102);">WebViewJSHelper</span>()<<span class="hljs-title" style="box-sizing: border-box; color: rgb(102, 0, 102);">WebViewJSHelperProtocol</span>></span> <span class="hljs-keyword" style="color: rgb(0, 0, 136); box-sizing: border-box;">@end</span> <span class="hljs-class" style="box-sizing: border-box;"><span class="hljs-keyword" style="color: rgb(0, 0, 136); box-sizing: border-box;">@implementation</span> <span class="hljs-title" style="box-sizing: border-box; color: rgb(102, 0, 102);">WebViewJSHelper</span></span> -(<span class="hljs-keyword" style="color: rgb(0, 0, 136); box-sizing: border-box;">void</span>)log:(<span class="hljs-built_in" style="color: rgb(102, 0, 102); box-sizing: border-box;">NSString</span>*)arg{ <span class="hljs-built_in" style="color: rgb(102, 0, 102); box-sizing: border-box;">NSLog</span>(@<span class="hljs-string" style="color: rgb(0, 136, 0); box-sizing: border-box;">"%@"</span>,arg); } <span class="hljs-keyword" style="color: rgb(0, 0, 136); box-sizing: border-box;">@end</span></code><ul class="pre-numbering" style="box-sizing: border-box; position: absolute; width: 50px; background-color: rgb(238, 238, 238); top: 0px; left: 0px; margin: 0px; padding: 6px 0px 40px; border-right-width: 1px; border-right-style: solid; border-right-color: rgb(221, 221, 221); list-style: none; text-align: right;"><li style="box-sizing: border-box; padding: 0px 5px;">1</li><li style="box-sizing: border-box; padding: 0px 5px;">2</li><li style="box-sizing: border-box; padding: 0px 5px;">3</li><li style="box-sizing: border-box; padding: 0px 5px;">4</li><li style="box-sizing: border-box; padding: 0px 5px;">5</li><li style="box-sizing: border-box; padding: 0px 5px;">6</li><li style="box-sizing: border-box; padding: 0px 5px;">7</li><li style="box-sizing: border-box; padding: 0px 5px;">8</li></ul>
3.注册该类的实例到JSContext
<code class="hljs bash has-numbering" style="display: block; padding: 0px; background-color: transparent; color: inherit; box-sizing: border-box; font-family: 'Source Code Pro', monospace;font-size:undefined; white-space: pre; border-top-left-radius: 0px; border-top-right-radius: 0px; border-bottom-right-radius: 0px; border-bottom-left-radius: 0px; word-wrap: normal; background-position: initial initial; background-repeat: initial initial;">WebViewJSHelper *_webViewJSHelper = [[WebViewJSHelper alloc]init]; _jsContext[@<span class="hljs-string" style="color: rgb(0, 136, 0); box-sizing: border-box;">"xxname"</span>] = _webViewJSHelper;</code><ul class="pre-numbering" style="box-sizing: border-box; position: absolute; width: 50px; background-color: rgb(238, 238, 238); top: 0px; left: 0px; margin: 0px; padding: 6px 0px 40px; border-right-width: 1px; border-right-style: solid; border-right-color: rgb(221, 221, 221); list-style: none; text-align: right;"><li style="box-sizing: border-box; padding: 0px 5px;">1</li><li style="box-sizing: border-box; padding: 0px 5px;">2</li></ul>
到此,JS调用xxname.log方法时,会进入我们iOS端定义的log方法中执行。
关于JSExport还有一些定义别名之类的,感兴趣的同学可以自行看API或网上文档。
方案二:
这个方案的操作方式就简单多了,对比方案一,无需定义实现的原生方法,操作如下:
1.随便定义一个类加上JSExport协议
<code class="hljs objectivec has-numbering" style="display: block; padding: 0px; background-color: transparent; color: inherit; box-sizing: border-box; font-family: 'Source Code Pro', monospace;font-size:undefined; white-space: pre; border-top-left-radius: 0px; border-top-right-radius: 0px; border-bottom-right-radius: 0px; border-bottom-left-radius: 0px; word-wrap: normal; background-position: initial initial; background-repeat: initial initial;"><span class="hljs-class" style="box-sizing: border-box;"><span class="hljs-keyword" style="color: rgb(0, 0, 136); box-sizing: border-box;">@interface</span> <span class="hljs-title" style="box-sizing: border-box; color: rgb(102, 0, 102);">WebViewJSExport</span>()<<span class="hljs-title" style="box-sizing: border-box; color: rgb(102, 0, 102);">JSExport</span>></span> <span class="hljs-keyword" style="color: rgb(0, 0, 136); box-sizing: border-box;">@end</span> <span class="hljs-class" style="box-sizing: border-box;"><span class="hljs-keyword" style="color: rgb(0, 0, 136); box-sizing: border-box;">@implementation</span> <span class="hljs-title" style="box-sizing: border-box; color: rgb(102, 0, 102);">WebViewJSExport</span></span> <span class="hljs-keyword" style="color: rgb(0, 0, 136); box-sizing: border-box;">@end</span></code><ul class="pre-numbering" style="box-sizing: border-box; position: absolute; width: 50px; background-color: rgb(238, 238, 238); top: 0px; left: 0px; margin: 0px; padding: 6px 0px 40px; border-right-width: 1px; border-right-style: solid; border-right-color: rgb(221, 221, 221); list-style: none; text-align: right;"><li style="box-sizing: border-box; padding: 0px 5px;">1</li><li style="box-sizing: border-box; padding: 0px 5px;">2</li><li style="box-sizing: border-box; padding: 0px 5px;">3</li><li style="box-sizing: border-box; padding: 0px 5px;">4</li></ul>
2.注册给JSContext
<code class="hljs cs has-numbering" style="display: block; padding: 0px; background-color: transparent; color: inherit; box-sizing: border-box; font-family: 'Source Code Pro', monospace;font-size:undefined; white-space: pre; border-top-left-radius: 0px; border-top-right-radius: 0px; border-bottom-right-radius: 0px; border-bottom-left-radius: 0px; word-wrap: normal; background-position: initial initial; background-repeat: initial initial;">_jsContext[<span class="hljs-string" style="color: rgb(0, 136, 0); box-sizing: border-box;">@"xxname"</span>] = [WebViewJSExport <span class="hljs-keyword" style="color: rgb(0, 0, 136); box-sizing: border-box;">new</span>];</code><ul class="pre-numbering" style="box-sizing: border-box; position: absolute; width: 50px; background-color: rgb(238, 238, 238); top: 0px; left: 0px; margin: 0px; padding: 6px 0px 40px; border-right-width: 1px; border-right-style: solid; border-right-color: rgb(221, 221, 221); list-style: none; text-align: right;"><li style="box-sizing: border-box; padding: 0px 5px;">1</li></ul>
3.处理某个JS端调起的方法
<code class="hljs objectivec has-numbering" style="display: block; padding: 0px; background-color: transparent; color: inherit; box-sizing: border-box; font-family: 'Source Code Pro', monospace;font-size:undefined; white-space: pre; border-top-left-radius: 0px; border-top-right-radius: 0px; border-bottom-right-radius: 0px; border-bottom-left-radius: 0px; word-wrap: normal; background-position: initial initial; background-repeat: initial initial;"><span class="hljs-comment" style="color: rgb(136, 0, 0); box-sizing: border-box;">//定义不含xxname的正常js方法</span> _jsContext[@<span class="hljs-string" style="color: rgb(0, 136, 0); box-sizing: border-box;">"log"</span>] = ^() { <span class="hljs-built_in" style="color: rgb(102, 0, 102); box-sizing: border-box;">NSArray</span> *args = [JSContext currentArguments]; <span class="hljs-keyword" style="color: rgb(0, 0, 136); box-sizing: border-box;">for</span> (<span class="hljs-keyword" style="color: rgb(0, 0, 136); box-sizing: border-box;">id</span> obj in args) { <span class="hljs-comment" style="color: rgb(136, 0, 0); box-sizing: border-box;">//需要注意这里的obj都还是JSValue</span> <span class="hljs-built_in" style="color: rgb(102, 0, 102); box-sizing: border-box;">NSLog</span>(@<span class="hljs-string" style="color: rgb(0, 136, 0); box-sizing: border-box;">"%@"</span>,obj); } }; <span class="hljs-comment" style="color: rgb(136, 0, 0); box-sizing: border-box;">//重定义xxname.log方法</span> [_jsContext evaluateScript:@<span class="hljs-string" style="color: rgb(0, 136, 0); box-sizing: border-box;">"xxname.log = log"</span>];</code><ul class="pre-numbering" style="box-sizing: border-box; position: absolute; width: 50px; background-color: rgb(238, 238, 238); top: 0px; left: 0px; margin: 0px; padding: 6px 0px 40px; border-right-width: 1px; border-right-style: solid; border-right-color: rgb(221, 221, 221); list-style: none; text-align: right;"><li style="box-sizing: border-box; padding: 0px 5px;">1</li><li style="box-sizing: border-box; padding: 0px 5px;">2</li><li style="box-sizing: border-box; padding: 0px 5px;">3</li><li style="box-sizing: border-box; padding: 0px 5px;">4</li><li style="box-sizing: border-box; padding: 0px 5px;">5</li><li style="box-sizing: border-box; padding: 0px 5px;">6</li><li style="box-sizing: border-box; padding: 0px 5px;">7</li><li style="box-sizing: border-box; padding: 0px 5px;">8</li><li style="box-sizing: border-box; padding: 0px 5px;">9</li><li style="box-sizing: border-box; padding: 0px 5px;">10</li></ul>
好了,现在整套功能都可以使用了。
上面方案二的方式个人觉得优点更多一些,首先更贴近JS习惯,其次不需要定义那么多的方法,而且对于js端调用来说,不定参数个数也更容易些。
这篇文章的最后,讲一点小的注意事项:
1. 使用 NSArray *args = [JSContext currentArguments];得到的args,内部都是JSValue,千万注意这一点,如果想要使用,最好转成相应的类型,比如[obj toString]之类的,具体可以看JSValue的头文件
2. 要注意循环引用的问题,不要写出 _jsContext[@”xxname”] = self; 这样的代码来,其他的就是block中的了,老生常谈不再细说。
3. 要注意JS端和原生端生命周期的问题
两端的内存管理方式不一样,JS端是GC,我们是引用计数,一般情况不会有生命周期的问题,但是必要时,需要使用JSVirtualMachine包装一下,这块我也还在学习,希望大家注意。
相关文章推荐
- [Android学习笔记十]Adapter如何进行视图与数据绑定资料
- Android自定义控件--圆形头像
- android屏幕适配,除了使用dp,还可以使用比例
- Android耗电评估
- Android Studio无法在线更新
- Android中Activity继承思想
- Android开发相关的Blog推荐——跟随大神的脚步才能成长为大神
- Android 代码设置 控件背景颜色
- Android图表生成类库Demo
- Android MediaPlayer 播放prepareAsync called in state 8解决办法
- Android SDK Manager无法下载其他版本的API 解决办法
- 【Android实测】CountDownTimer类的使用
- android jni
- Android .9.png图片制作
- Android MediaStore扫描 & 向MediaStore中插入文件记录
- Android ActionBar 基本使用
- Activity的生命周期
- Android 切.9图
- [Android]TextView设置字体大小时应该知道的事-同样的textSize不同的效果
- Android Studio 下使用NDK