說說JavaScriptCore

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.js
  var 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函數,有點方便~
    JSContext *context = [[JSContext alloc] init];
    context[@"sayhi"] = ^(NSString *name) {
        NSLog(@"say hi to %@",name);
    };
    [context evaluateScript:@"sayhi('Greg')"]; //"say hi to Greg"
block.png
  • JSExport協議
    如果你到頭文件中去查看JSExport協議,你會發現這個協議其實沒有定義任何東西。JavaScriptCore提供這個協議用來將oc的方法跟屬性暴露給js調用,其中@property會轉換成js的getter和setter方法,實例方法會轉換成js函數,而類方法則轉換成js中global object的方法。
JSExport

舉個例子簡單說明一下,我們的PersonProtocol協議定義好要暴露給js的內容:

//定義需要暴露給js的內容,這里我們只暴露personName和queryPersonName接口
@protocol PersonProtocol <JSExport>
@property(nonatomic,copy)NSString *personName;
-(NSString *)queryPersonName;
@end

//Person實現PersonProtocol協議,而自己定義的age和queryPersonAge接口不暴露給js
@interface Person : NSObject <PersonProtocol>
@property(nonatomic,assign)NSInteger age;
-(NSInteger)queryPersonAge;
@end

@implementation Person
@synthesize personName = _personName;

-(NSString *)queryPersonName{
    return self.personName;
}
-(NSInteger)queryPersonAge{
    return self.age;
}
@end

JSContext *context = [[JSContext alloc] init];

//創建Person類的對象,將他賦值給js對象
Person *person=[Person new];
person.personName = @"Greg";
person.age = 27;
context[@"person"]=person;

//可以調用獲取PersonProtocol暴露的內容
NSString *personName = [[context evaluateScript:@"person.personName"] toString]; //"Greg"
NSString *personName1 = [[context evaluateScript:@"person.queryPersonName()"] toString]; //"Greg"

//js無法調用跟age相關的內容
NSInteger age = [[context evaluateScript:@"person.age"] toInt32]; // 0
NSInteger age1 = [[context evaluateScript:@"person.queryPersonAge()"] toInt32]; //0

內存管理

下面我們來說說內存管理方面的問題,我們知道在oc中使用ARC方式管理內存(基于引用計數),但JavaScriptCore中使用的是垃圾回收方式,其中所有的引用都是強引用,我們不必擔心其循環引用,js的垃圾回收能夠打破這些強引用,通常我們在使用JavaScriptCore中的API時不太需要去關注內存問題,因為這些都會被自動處理好,不過有些情況需要我們注意一下:

  • 在oc對象中存儲js的值
    如果在oc對象中存儲js的值,需要注意一下不要導致循環引用,看個例子:
//test.js
function ClickHandler(button, callback) {
  this.button = button;
  this.handler = callback;
}
//test.m
@implement MyButton
-(void)setClickHandler:(JSValue*)handler
{
  _onClickHandler = handler; //導致retain cycle
}

上面的代碼中可以看出,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:
//bad
JSContext *context = [[JSContext alloc] init];
context[@"callback"] = ^{
     JSValue *object = [JSValue valueWithNewObjectInContext:context];
     return object;
};

//good
JSContext *context = [[JSContext alloc] init];
context[@"callback"] = ^{
     JSValue *object = [JSValue valueWithNewObjectInContext:
        [JSContext currentContext]];
     return object;
};

多線程

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/

最后編輯于
?著作權歸作者所有,轉載或內容合作請聯系作者
平臺聲明:文章內容(如有圖片或視頻亦包括在內)由作者上傳并發布,文章內容僅代表作者本人觀點,簡書系信息發布平臺,僅提供信息存儲服務。
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市,隨后出現的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 228,345評論 6 531
  • 序言:濱河連續發生了三起死亡事件,死亡現場離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機,發現死者居然都...
    沈念sama閱讀 98,494評論 3 416
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人,你說我怎么就攤上這事。” “怎么了?”我有些...
    開封第一講書人閱讀 176,283評論 0 374
  • 文/不壞的土叔 我叫張陵,是天一觀的道長。 經常有香客問我,道長,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 62,953評論 1 309
  • 正文 為了忘掉前任,我火速辦了婚禮,結果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當我...
    茶點故事閱讀 71,714評論 6 410
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發上,一...
    開封第一講書人閱讀 55,186評論 1 324
  • 那天,我揣著相機與錄音,去河邊找鬼。 笑死,一個胖子當著我的面吹牛,可吹牛的內容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 43,255評論 3 441
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了?” 一聲冷哼從身側響起,我...
    開封第一講書人閱讀 42,410評論 0 288
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后,有當地人在樹林里發現了一具尸體,經...
    沈念sama閱讀 48,940評論 1 335
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內容為張勛視角 年9月15日...
    茶點故事閱讀 40,776評論 3 354
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發現自己被綠了。 大學時的朋友給我發了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 42,976評論 1 369
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 38,518評論 5 359
  • 正文 年R本政府宣布,位于F島的核電站,受9級特大地震影響,放射性物質發生泄漏。R本人自食惡果不足惜,卻給世界環境...
    茶點故事閱讀 44,210評論 3 347
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 34,642評論 0 26
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春,著一層夾襖步出監牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 35,878評論 1 286
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個月前我還...
    沈念sama閱讀 51,654評論 3 391
  • 正文 我出身青樓,卻偏偏與公主長得像,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 47,958評論 2 373

推薦閱讀更多精彩內容