JavaScriptCore 使用
2015-12-10 15:04
639 查看
JavaScriptCore
JavaScriptCore是webkit的一个重要组成部分,主要是对JS进行解析和提供执行环境。代码是开源的,可以下下来看看(源码)。iOS7后苹果在iPhone平台推出,极大的方便了我们对js的操作。我们可以脱离webview直接运行我们的js。iOS7以前我们对JS的操作只有webview里面一个函数 stringByEvaluatingJavaScriptFromString,JS对OC的回调都是基于URL的拦截进行的操作。大家用得比较多的是WebViewJavascriptBridge和EasyJSWebView这两个开源库,很多混合都采用的这种方式。
JavaScriptCore和我们相关的类不是很多,使用起来也非常简单。
#import "JSContext.h" #import "JSValue.h" #import "JSManagedValue.h" #import "JSVirtualMachine.h" #import "JSExport.h"
JSContext
JS执行的环境,同时也通过JSVirtualMachine管理着所有对象的生命周期,每个JSValue都和JSContext相关联并且强引用context。
JSValue
JS对象在JSVirtualMachine中的一个强引用,其实就是Hybird对象。我们对JS的操作都是通过它。并且每个JSValue都是强引用一个context。同时,OC和JS对象之间的转换也是通过它,相应的类型转换如下:Objective-C type | JavaScript type --------------------+--------------------- nil | undefined NSNull | null NSString | string NSNumber | number, boolean NSDictionary | Object object NSArray | Array object NSDate | Date object NSBlock (1) | Function object (1) id (2) | Wrapper object (2) Class (3) | Constructor object (3)
JSManagedValue
JS和OC对象的内存管理辅助对象。由于JS内存管理是垃圾回收,并且JS中的对象都是强引用,而OC是引用计数。如果双方相互引用,势必会造成循环引用,而导致内存泄露。我们可以用JSManagedValue保存JSValue来避免。
JSVirtualMachine
JS运行的虚拟机,有独立的堆空间和垃圾回收机制。
JSExport
一个协议,如果JS对象想直接调用OC对象里面的方法和属性,那么这个OC对象只要实现这个JSExport协议就可以了。
OC和JS之间的通信
两者之间的通信还是很简单的,直接看简单代码示例吧。
Objective-C -> JavaScript
self.context = [[JSContext alloc] init]; NSString *js = @"function add(a,b) {return a+b}"; [self.context evaluateScript:js]; JSValue *n = [self.context[@"add"] callWithArguments:@[@2, @3]]; NSLog(@"---%@", @([n toInt32]));//---5
步骤很简单,创建一个JSContext对象,然后将JS代码加载到context里面,最后取到这个函数对象,调用
callWithArguments这个方法进行参数传值。(JS里面函数也是对象)
JavaScript -> Objective-C
JS调用OC有两个方法:block和JSExport protocol。block(JS function):
self.context = [[JSContext alloc] init]; self.context[@"add"] = ^(NSInteger a, NSInteger b) { NSLog(@"---%@", @(a + b)); }; [self.context evaluateScript:@"add(2,3)"];
我们定义一个block,然后保存到context里面,其实就是转换成了JS的function。然后我们直接执行这个function,调用的就是我们的block里面的内容了。
JSExport protocol:
//定义一个JSExport protocol @protocol JSExportTest <JSExport> - (NSInteger)add:(NSInteger)a b:(NSInteger)b; @property (nonatomic, assign) NSInteger sum; @end //建一个对象去实现这个协议: @interface JSProtocolObj : NSObject<JSExportTest> @end @implementation JSProtocolObj @synthesize sum = _sum; //实现协议方法 - (NSInteger)add:(NSInteger)a b:(NSInteger)b { return a+b; } //重写setter方法方便打印信息, - (void)setSum:(NSInteger)sum { NSLog(@"--%@", @(sum)); _sum = sum; } @end //在VC中进行测试 @interface ViewController () <JSExportTest> @property (nonatomic, strong) JSProtocolObj *obj; @property (nonatomic, strong) JSContext *context; @end @implementation ViewController - (void)viewDidLoad { [super viewDidLoad]; //创建context self.context = [[JSContext alloc] init]; //设置异常处理 self.context.exceptionHandler = ^(JSContext *context, JSValue *exception) { [JSContext currentContext].exception = exception; NSLog(@"exception:%@",exception); }; //将obj添加到context中 self.context[@"OCObj"] = self.obj; //JS里面调用Obj方法,并将结果赋值给Obj的sum属性 [self.context evaluateScript:@"OCObj.sum = OCObj.addB(2,3)"]; }
demo很简单,还是定义了一个两个数相加的方法,还有一个保存结果的变量。在JS中进行调用这个对象的方法,并将结果赋值sum。唯一要注意的是OC的函数命名和JS函数命名规则问题。协议中定义的是
add: b:,但是JS里面方法名字是
addB(a,b)。可以通过
JSExportAs这个宏转换成JS的函数名字。
修改下代码:
@protocol JSExportTest <JSExport> //用宏转换下,将JS函数名字指定为add; JSExportAs(add, - (NSInteger)add:(NSInteger)a b:(NSInteger)b); @property (nonatomic, assign) NSInteger sum; @end //调用 [self.context evaluateScript:@"OCObj.sum = OCObj.add(2,3)"];
我们可以定义自己的异常捕获,可以把context,异常block改为自己的:
self.context.exceptionHandler = ^(JSContext *context, JSValue *exception) { [JSContext currentContext].exception = exception; NSLog(@"exception:%@",exception); };
相关文章推荐
- 转----在.NET使用JSON作为数据交换格式
- [Javascript] Ajax
- js学习笔记 - String字符串对象
- 【JavaScript】网页登录程序之验证码篇
- js数组去重 javascript版
- JSON序列化与反序列化JavaScript
- 通过 JSON 字符串来创建对象&通过eval()方法生成原生Javascript对象
- <html5+css3+js>一个audio note界面
- 初识backbone.js(1)
- Javascript常见操作
- JSON 中的毫秒时间( LONG) ,转换成年月日
- JavaScript 闭包
- js--在页面元素上(移动到或获取焦点)、鼠标离开(或失去焦点)
- gulpfile.js 中的坑--注入篇(gulp-inject,gulp-wiredep)
- js取json数据笔记
- 解决方案:Jboss修改jsp需要重启才生效
- JSON序列化与反序列化DataSet
- 【JavaScript 从零开始】 语言核心部分----可选的分号
- js unicode编码与解码
- js操作数组函数实例小结