JavaScriptCore詳解

本博客主要分以下幾個方面來介紹iOS中的JavaScriptCore

  • JavaScriptCore簡介
  • JavaScriptCore中主要的類
  • JSContext
  • JSValue
  • JSExport
  • JSManagedValue
  • JSVirtualMachine
  • Native Code 和 JS 之間的互相調用
  • Native Code 與UIWebView中的JS交互
  • Native Code 與JS文件直接交互

JavaScriptCore簡介

JavaScriptCore背景

  • iOS中的JavaScriptCore.framework其實只是基于webkit(Safari的瀏覽器引擎)中以C/C++實現的JavaScriptCore的一個包裝,在iOS7中,Apple將其作為一個標準庫供開發者使用

JavaScriptCore主要功能

  1. JavaScriptCore主要是對JS進行解析和提供執行環境。代碼是開源的,JavaScriptCore源碼
  2. JavaScriptCore可以讓我們脫離webview直接運行我們的js
  3. JavaScriptCore提供一種動態局部升級和更新的邏輯,大大提高應用的可擴展性
  4. 對手機內嵌web模式的新嘗試點,即通過Native+JS file的方式取代webview的方式

JavaScriptCore中主要的類

  1. JSContext --- 在OC中創建JavaScript運行的上下文環境

    - (instancetype)init; // 創建JSContext對象,獲得JavaScript運行的上下文環境
    
    // 在特定的對象空間上創建JSContext對象,獲得JavaScript運行的上下文環境
    - (instancetype)initWithVirtualMachine:(JSVirtualMachine *)virtualMachine;
    
    // 運行一段js代碼,輸出結果為JSValue類型
    - (JSValue *)evaluateScript:(NSString *)script;
    
    // iOS 8.0以后可以調用此方法
    - (JSValue *)evaluateScript:(NSString *)script withSourceURL:(NSURL *)sourceURL NS_AVAILABLE(10_10, 8_0);
    
    // 獲取當前正在運行的JavaScript上下文環境
    + (JSContext *)currentContext;
    
    // 返回結果當前執行的js函數 function () { [native code] } ,iOS 8.0以后可以調用此方法
     + (JSValue *)currentCallee NS_AVAILABLE(10_10, 8_0);
    
     // 返回結果當前方法的調用者[object Window]
     + (JSValue *)currentThis;
    
     // 返回結果為當前被調用方法的參數
     + (NSArray *)currentArguments;
    
     // js的全局變量 [object Window]
     @property (readonly, strong) JSValue *globalObject;
    
  2. JSValue --- JavaScript中的變量和方法,可以轉成OC數據類型,每個JSValue都和JSContext相關聯并且強引用context

         @textblock
        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)
         @/textblock
     // 在context創建BOOL的JS變量
     + (JSValue *)valueWithBool:(BOOL)value inContext:(JSContext *)context;
    
     // 將JS變量轉換成OC中的BOOL類型
     - (BOOL)toBool;
    
     // 修改JS對象的屬性的值
     - (void)setValue:(id)value forProperty:(NSString *)property;
    
     // JS中是否有這個對象
     @property (readonly) BOOL isUndefined;
    
     // 比較兩個JS對象是否相等
     - (BOOL)isEqualToObject:(id)value;
    
     // 調用者JSValue對象為JS中的方法名稱,arguments為參數,調用JS中Window直接調用的方法
     - (JSValue *)callWithArguments:(NSArray *)arguments;
    
    // 調用者JSValue對象為JS中的全局對象名稱,method為全局對象的方法名稱,arguments為參數
     - (JSValue *)invokeMethod:(NSString *)method withArguments:(NSArray *)arguments;
    
    // JS中的結構體類型轉換為OC
     + (JSValue *)valueWithPoint:(CGPoint)point inContext:(JSContext *)context;
    
  3. JSExport --- JS調用OC中的方法和屬性寫在繼承自JSExport的協議當中,OC對象實現自定義的協議

    // textFunction -- JS方法
    // - (void) ocTestFunction:(NSNumber *)value sec:(NSNumber *)number -- OC方法
    JSExportAs (textFunction,- (void) ocTestFunction:(NSNumber *)value sec:(NSNumber *)number);
    
  4. JSManagedValue --- JS和OC對象的內存管理輔助對象,主要用來保存JSValue對象,解決OC對象中存儲js的值,導致的循環引用問題

    JSManagedValue *_jsManagedValue = [JSManagedValue managedValueWithValue:jsValue];
    [_context.virtualMachine addManagedReference:_jsManagedValue];
    

    JSManagedValue本身只弱引用js值,需要調用JSVirtualMachine的addManagedReference:withOwner:把它添加到JSVirtualMachine中,這樣如果JavaScript能夠找到該JSValue的Objective-C owner,該JSValue的引用就不會被釋放。

  5. JSVirtualMachine --- JS運行的虛擬機,有獨立的堆空間和垃圾回收機制,運行在不同虛擬機環境的JSContext可以通過此類通信。

Native Code 和 JS 之間的互相調用(以UIWebView中的JS為例)

  1. JS中,點擊事件直接調用方法方式JS和Native代碼的互調如下:
  • 1.1 JS代碼如下

    <html>
      <meta http-equiv="Content-Type" content="text/html; charset=utf-8" />
      <meta content="width=device-width,initial-scale=1,user-scalable=no" name="viewport">
    <body>
      <script type="text/javascript">
      var nativeCallJS = function(parameter) {
       alert (parameter);
      };
      </script>
      <button type="button" onclick = "jsCallNative('jsParameter')" style="width:100%; height:30px;"/>調用OC代碼</button>
    </body>
    </html>
    
    
  • 1.2 OC代碼如下

      - (void)__jsLogic
     {
         self.jsContext.exceptionHandler = ^(JSContext *context, JSValue *exception){
             NSLog(@"JS代碼執行中的異常信息%@", exception);
         };
         self.jsContext = [self valueForKeyPath:@"documentView.webView.mainFrame.javaScriptContext"];
         self.jsContext[@"jsCallNative"] = ^(NSString *paramer){
             JSValue *currentThis = [JSContext currentThis];
             JSValue *currentCallee = [JSContext currentCallee];
             NSArray *currentParamers = [JSContext currentArguments];
             dispatch_async(dispatch_get_main_queue(), ^{
                   /**
                    *  js調起OC代碼,代碼在子線程,更新OC中的UI,需要回到主線程
                    */
             });
             NSLog(@"JS paramer is %@",paramer);
             NSLog(@"currentThis is %@",[currentThis toString]);
             NSLog(@"currentCallee is %@",[currentCallee toString]);
             NSLog(@"currentParamers is %@",currentParamers);
         };
         JSValue *jsMethod = self.jsContext[@"nativeCallJS"];
         [jsMethod callWithArguments:@[@"nativeCallJS"]];
     }
    
  • 1.3 OC運行結果,彈出HTML中的alert提示

    2016-08-05 17:57:08.974 LeWebViewPro[38150:3082770] JS paramer is jsParameter
    2016-08-05 17:57:08.975 LeWebViewPro[38150:3082770] currentThis is [object Window]
    2016-08-05 17:57:08.975 LeWebViewPro[38150:3082770] currentCallee is function () {
    [native code]
    }
    2016-08-05 17:57:08.975 LeWebViewPro[38150:3082770] currentParamers is (
    jsParameter
    )
    
  1. JS中,點擊事件等事件通過調用全局對象的方法調用方法,JS和Native代碼的互調如下:
    • 2.1 JS代碼如下

      <html>
        <meta http-equiv="Content-Type" content="text/html; charset=utf-8" />
        <meta content="width=device-width,initial-scale=1,user-scalable=no" name="viewport">
      <body>
        <script type="text/javascript">
        globalObject = new Object();
        globalObject.name = 100;
        globalObject.nativeCallJS = function (parameter) {
         alert (parameter);
        };
        </script>
        <button type="button" onclick = "globalObject.jsCallNative('jsParameter')" style="width:100%; height:30px;"/>調用OC代碼</button>
      </body>
      </html>
      
      
    • 2.2 OC代碼如下

    • 2.2.1 JSManager 代碼,負責執行JS中的方法

      #import <JavaScriptCore/JavaScriptCore.h>
      #import <Foundation/Foundation.h>
      @protocol LeJSExport <JSExport>
      JSExportAs (jsCallNative,- (void) jsCallNative:(NSString *)jsParameter);
      @end
      
      @interface JSManager : NSObject<LeJSExport>
      
      @end
      -----.M文件-----
      #import "JSManager.h"
      @implementation JSManager
      - (void)jsCallNative:(NSString *)jsParameter
      {
         JSValue *currentThis = [JSContext currentThis];
         JSValue *currentCallee = [JSContext currentCallee];
         NSArray *currentParamers = [JSContext currentArguments];
         dispatch_async(dispatch_get_main_queue(), ^{
             /**
              *  js調起OC代碼,代碼在子線程,更新OC中的UI,需要回到主線程
              */
         });
         NSLog(@"JS paramer is %@",jsParameter);
         NSLog(@"currentThis is %@",[currentThis toString]);
         NSLog(@"currentCallee is %@",[currentCallee toString]);
         NSLog(@"currentParamers is %@",currentParamers);
      }
      @end
      
    • 2.2.2 包含UIWebView類的代碼,負責調起JS中的方法

      - (void)nativeCallJS
      {
          self.jsManager = [[JSManager alloc] init];
          self.jsContext = [self valueForKeyPath:@"documentView.webView.mainFrame.javaScriptContext"];
          self.jsContext.exceptionHandler = ^(JSContext *context, JSValue *exception){
              NSLog(@"JS代碼執行中的異常信息%@", exception);
          };
          self.jsContext[@"globalObject"] = self.jsManager;
          //1.OC方法調起JS
          JSValue *varibleStyle = self.jsContext[@"globalObject"];
          [varibleStyle invokeMethod:@"nativeCallJS" withArguments:@[@100]];
      
          //2.OC腳本調起JS
          NSString *jsScript = [NSString stringWithFormat:@"globalObject.nativeCallJS('%@')",@100];
          [self.jsContext evaluateScript:jsScript];
      }
      
    • 2.3 OC運行結果,彈出HTML中的alert提示

      2016-08-07 10:30:11.444 LeWebViewPro[46674:3253852] JS paramer is jsParameter
      2016-08-07 10:30:11.444 LeWebViewPro[46674:3253852] currentThis is [object JSManager]
      2016-08-07 10:30:11.444 LeWebViewPro[46674:3253852] currentCallee is function () {
          [native code]
      }
      2016-08-07 10:30:11.445 LeWebViewPro[46674:3253852] currentParamers is (
          jsParameter
      )
      

JavaScriptCore和UIWebView的使用的注意事項

  1. OC在與UIWebView中的JS交互的邏輯是,先獲取UIWebView中的JS的執行環境

    self.context = [webView valueForKeyPath:@"documentView.webView.mainFrame.javaScriptContext"];
    
  2. 獲取UIWebView中的JS的執行環境的時機,一般在webViewDidFinishLoad時獲取,獲取不到的情況下,需改在其他方法中獲取

    shouldStartLoadWithRequest:Sent before a web view begins loading a frame
    webViewDidStartLoad:Sent after a web view starts loading a frame.
    webViewDidFinishLoad:Sent after a web view finishes loading a frame
    
  3. 線程問題

    • JavaScriptCore中提供的API都是線程安全的,一個JSVirtualMachine在一個線程中,它可以包含多個JSContext,而且相互之間可以傳值,為了確保線程安全,這些context在運行的時候會采用鎖,可以認為是串行執行。
    • JS調用OC的回調方法,是在子線程,所以需要更新OC中的UI的話,需要切換到主線程
  4. 內存問題

    • oc中使用ARC方式管理內存(基于引用計數),但JavaScriptCore中使用的是垃圾回收方式,其中所有的引用都是強引用,但是我們不必擔心其循環引用,js的垃圾回收能夠打破這些強引用,有些情況需要考慮如下

    • js調起OC回調的block中獲取JSConetxt容易循環引用

      self.jsContext[@"jsCallNative"] = ^(NSString *paramer){
        // 會引起循環引用
        JSValue *value1 =  [JSValue valueWithNewObjectInContext:
                            self.jsContext];
        // 不會引起循環引用
        JSValue *value =  [JSValue valueWithNewObjectInContext:
                           [JSContext currentContext]];
      
      };
      
    
    * JavaScriptCore中所有的引用都是強引用,所以在OC中需要存儲JS中的值的時候,需要注意
    >在oc中為了打破循環引用我們采用weak的方式,不過在JavaScriptCore中我們采用內存管理輔助對象JSManagedValue的方式,它能幫助引用技術和垃圾回收這兩種內存管理機制之間進行正確的轉換
    
    

JavaScriptCore單獨使用

  1. js代碼如下test.js
globalObject = new Object();
globalObject.name = 100;
globalObject.nativeCallJS = function (parameter) {
    alert (parameter);
};
  1. OC讀取JS文件,并相互通信
NSString *jsPath = [[NSBundle mainBundle] pathForResource:@"test" ofType:@"js"];
    NSString *jsContent = [[NSString alloc] initWithContentsOfFile:jsPath encoding:NSUTF8StringEncoding error:nil];

    JSContext *jsContext = [[JSContext alloc] init];
    //捕獲運行js腳本的錯誤信息
    jsContext.exceptionHandler = ^(JSContext *context, JSValue *exceptionValue) {
        context.exception = exceptionValue;
        NSLog(@"異常信息:%@", exceptionValue);
    };
    //js腳本添加到當前的js執行環境中
    [jsContext evaluateScript:jsContent];
    self.jsManager = [[JSManager alloc] init];
    jsContext[@"globalObject"] = self.jsManager;
    ...
    ...
  1. jS與OC的交互與通過UIWebView相同

相關博客

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

推薦閱讀更多精彩內容