javascript
说说JavaScriptCore
http://www.jianshu.com/p/1328e15416f3/comments/1724404
javascript目前看來仍是世界上最流行的語言,不管在web、服務端還是客戶端都有廣泛的應用,很多跨平臺方案也采用js來實現,比如著名的reactjs,蘋果在iOS7引入了javascriptcore庫,提供更簡單方便的方式將js接入,iOS7之前要執行js操作只能通過UIWebview中的
stringByEvaluatingJavaScriptFromString方法,而且JavaScriptCore這塊的代碼開源,可以到這里查看,本文就簡單介紹一下JavaScriptCore:
- Objective-C調用JavaScript
- JavaScript調用Objective-C
- 內存管理
- 多線程
不過在那之前先介紹幾個基本概念:
- JSContext
一個JSContext實例代表著一個js運行時環境,js代碼都需要在一個context上下文內執行,而且JSContext還負責管理js虛擬機中所有對象的生命周期 - JSValue
表示一個JavaScript的實體,一個JSValue可以表示很多JavaScript原始類型例如boolean, integers, doubles,甚至包括對象和函數。我們對JS的操作都是通過它,并且每個JSValue都強引用一個context。同時,OC和JS對象之間的轉換也是通過它,相應的類型轉換如下:
JSValue類型轉換
- JSVirtualMachine
js代碼運行的虛擬機,提供JavaScriptCore執行需要的資源,有自己獨立的堆棧以及垃圾回收機制,而且通過鎖來實現線程安全,如果需要并發執行js代碼,可以創建不同的JSVirtualMachine虛擬機對象來實現;
Objective-C調用JavaScript
oc想要調用js代碼的話,先創建一個JSContext對象實例,接著通過evaluateScript加載js代碼到context對象中,然后獲取js對象,如果為js函數對象,通過callWithArguments調用該js函數,并且可以以數組的方式傳遞參數。
//test.jsvar appendString = function(name) { return 'string:' + name; }; var arr = [1, 2 , 'hello world']; //test.m NSString *jsPath = [[NSBundle mainBundle] pathForResource:@"test"ofType:@"js"]; NSString *jsContent = [NSString stringWithContentsOfFile:jsPath encoding:NSUTF8StringEncoding error:nil]; JSContext *context = [[JSContext alloc] init]; [context evaluateScript:jsContent]; JSValue *value = [context[@"appendString"] callWithArguments:@[@"hello"]]; JSValue *value1 = context[@"arr"]; NSLog(@"appendString:%@",[value toString] );//appendString:string:hello NSLog(@"arr:%@",[value1 toArray] ); // arr:( // 1, // 2, // "hello world" // )JavaScript調用Objective-C
js調用oc有兩種實現方式
- Blocks方式
我們可以通過block的方式將oc代碼暴露給js,JavaScriptCore會自動將oc block包裝在js函數中,我們就可以直接在js中調用該block函數,有點方便~
block.png
- JSExport協議
如果你到頭文件中去查看JSExport協議,你會發現這個協議其實沒有定義任何東西。JavaScriptCore提供這個協議用來將oc的方法跟屬性暴露給js調用,其中@property會轉換成js的getter和setter方法,實例方法會轉換成js函數,而類方法則轉換成js中global object的方法。
JSExport
舉個例子簡單說明一下,我們的PersonProtocol協議定義好要暴露給js的內容:
內存管理
下面我們來說說內存管理方面的問題,我們知道在oc中使用ARC方式管理內存(基于引用計數),但JavaScriptCore中使用的是垃圾回收方式,其中所有的引用都是強引用,我們不必擔心其循環引用,js的垃圾回收能夠打破這些強引用,通常我們在使用JavaScriptCore中的API時不太需要去關注內存問題,因為這些都會被自動處理好,不過有些情況需要我們注意一下:
- 在oc對象中存儲js的值
如果在oc對象中存儲js的值,需要注意一下不要導致循環引用,看個例子:
上面的代碼中可以看出,mybutton的onclickHandler強引用了js的handler,而js的button又強引用了mybutton,這就會導致retain cycle的問題:
retain cycle
在oc中為了打破循環引用我們采用weak的方式,不過在JavaScriptCore中我們采用內存管理輔助對象JSManagedValue的方式,它能幫助引用技術和垃圾回收這兩種內存管理機制之間進行正確的轉,所以我們可以采用如下方式:
@implement MyButton -(void)setClickHandler:(JSValue*)handler {_onClickHandler = [JSManagedValue managedValueWithValue:handler];[_context.virtualMachine addManagedReference:_onClickHandler]; }JSManagedValue本身只弱引用js值,需要調用JSVirtualMachine的addManagedReference:withOwner:把它添加到JSVirtualMachine中,這樣如果JavaScript能夠找到該JSValue的Objective-C owner,該JSValue的引用就不會被釋放。
- block中捕獲JSContexts
我們知道block會默認強引用它所捕獲的對象,如下代碼所示,如果block中直接使用context也會造成循環引用,這使用我們最好采用[JSContext currentContext]來獲取當前的JSContext:
多線程
JavaScriptCore中提供的API都是線程安全的,一個JSVirtualMachine在一個線程中,它可以包含多個JSContext,而且相互之間可以傳值,為了確保線程安全,這些context在運行的時候會采用鎖,可以認為是串行執行。
同一個虛擬機可以相互訪問
假如我們需要并發的執行js代碼,我們也可以在創建JSContext的時候也指定其所在的虛擬機,不同的虛擬機處于不同的線程中,但是如果在不同的 JSVirtualMachine,上下文并不能直接互相傳值,在使用的過程中需要注意一下。
JSVirtualMachine *vm1 = [JSVirtualMachine new]; JSContext *ctxA1 = [[JSContext alloc] initWithVirtualMachine:vm1]; JSContext *ctxA2 = [[JSContext alloc] initWithVirtualMachine:vm1]; JSVirtualMachine *vm2 = [JSVirtualMachine new]; JSContext *ctxB = [[JSContext alloc] initWithVirtualMachine:vm2];不同一個虛擬機不可以相互訪問
參考
https://developer.apple.com/videos/play/wwdc2013/615/
https://developer.apple.com/library/prerelease/ios/documentation/Carbon/Reference/WebKit_JavaScriptCore_Ref/
原文鏈接:http://www.jianshu.com/p/1328e15416f3/comments/1724404
著作權歸作者所有,轉載請聯系作者獲得授權,并標注“簡書作者”。
總結
以上是生活随笔為你收集整理的说说JavaScriptCore的全部內容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: ATS写文件
- 下一篇: NOIP2016提高组复赛解题报告