Objective-C與JavaScript交互的那些事

注:此文只現(xiàn)在只推薦需要適配iOS7的同學(xué)讀,如果已經(jīng)扔掉iOS7简烤,強烈建議換用WKWebView塘淑。已出WKWebView文章WKWebView使用及注意點(keng)

最近公司的運營瞎搞了個活動痒筒,其活動要服務(wù)端提供數(shù)據(jù)支持,web前端在微信公眾賬號內(nèi)作為主要的運營陣地肩狂,而iOSAndroid要提供相應(yīng)的入口及頁面進行配合姥饰。一個活動傻谁,動用了各個端的程序猿。而在這里面技術(shù)方面主要就是涉及到web端和服務(wù)端的交互列粪,web前端iOS审磁、Android的交互。本人作為一個iOS開發(fā)者岂座,今天就聊聊web态蒂、iOSAndroid三端的交互掺逼,其實在說明白一點就是方法的互相調(diào)用而已吃媒。這里主要講解iOSAndroid會稍微提一下吕喘,僅作參考赘那。

此篇文章的邏輯圖

圖0-0 此篇文章的邏輯圖

概述

iOS原生應(yīng)用和web頁面的交互大致上有這幾種方法iOS7之后的JavaScriptCore攔截協(xié)議氯质、第三方框架WebViewJavaScriptBridge募舟、iOS8之后的WKWebView在這里主要講解JavaScriptCore攔截協(xié)議這兩種辦法。WebViewJavaScriptBridge是基于攔截協(xié)議進行的封裝闻察。學(xué)習(xí)成本相對JavaScriptCore較高拱礁,使用也不如JavaScriptCore方便本文不做敘述琢锋。WKWebView是iOS8之后推出的,還沒有成為主流使用呢灶,所以本篇文章也不做詳細敘述吴超。

Objective-C執(zhí)行JavaScript代碼

相關(guān)方法

// UIWebView的方法
- (nullable NSString *)stringByEvaluatingJavaScriptFromString:(NSString *)script;

// JavaScriptCore中JSContext的方法
- (JSValue *)evaluateScript:(NSString *)script;
- (JSValue *)evaluateScript:(NSString *)script withSourceURL:(NSURL *)sourceURL

相關(guān)應(yīng)用

用這些方法去執(zhí)行大段的JavaScript代碼是沒什么必要的,但是有些小場景用起來還是比較順手和實用的鸯乃,列舉兩個例子作為參考:

// 獲取當(dāng)前頁面的title
NSString *title = [webview stringByEvaluatingJavaScriptFromString:@"document.title"];

// 獲取當(dāng)前頁面的url
NSString *url = [webview stringByEvaluatingJavaScriptFromString:@"document.location.href"];

JavaScriptCore

iOS7之后蘋果推出了JavaScriptCore這個框架鲸阻,從而讓web頁面和本地原生應(yīng)用交互起來非常方便,而且使用此框架可以做到Android那邊和iOS相對統(tǒng)一缨睡,web前端寫一套代碼就可以適配客戶端的兩個平臺鸟悴,從而減少了web前端的工作量。

web前端

在三端交互中奖年,web前端要強勢一些细诸,一切傳值、方法命名都按web前端開發(fā)人員來定義陋守,讓另外兩端去做適配震贵。在這里以調(diào)用攝像頭和分享為例來詳細講解,測試網(wǎng)頁代碼取名為test.html嗅义,其代碼內(nèi)容如下:

test.html代碼內(nèi)容
<!DOCTYPE html>
<html>
<head lang="en">
    <meta charset="UTF-8">
</head>
<body>
    <div style="margin-top: 100px">
        <h1>Objective-C和JavaScript交互的那些事</h1>
        <input type="button" value="CallCamera" onclick="Toyun.callCamera()">
    </div>       
        
    <div>
        <input type="button" value="Share" onclick="callShare()">
    </div>

<script>
    var callShare = function() {
        var shareInfo = JSON.stringify({"title": "標(biāo)題", "desc": "內(nèi)容", "shareUrl": "http://www.reibang.com/p/f896d73c670a",
        "shareIco":"http://upload-images.jianshu.io/upload_images/1192353-fd26211d54aea8a9.png?imageMogr2/auto-orient/strip%7CimageView2/2/w/1240"});
        Toyun.share(shareInfo);
    }

    var picCallback = function(photos) {
        alert(photos);
    }

    var shareCallback = function(){
        alert('success');
    }
</script>
</body>
</html>
test.html代碼解釋

可能有些同學(xué)對web前端的一些知識不太熟悉屏歹,稍微對這段代碼做下解釋,先說ToyuniOSAndroid這兩邊在本地要注入的一個對象【參考下面iOS的代碼更容易明白】之碗,充當(dāng)原生應(yīng)用和web頁面之間的一個橋梁蝙眶。頁面上定義了兩個按鈕名字分別為CallCameraShare。點擊CallCamera會通過Toyun這個橋梁調(diào)用本地應(yīng)用的方法- (void)callCamera褪那,沒有傳參幽纷;而點擊Share會先調(diào)用本文件中的JavaScript方法callShare這里將要分享的內(nèi)容格式轉(zhuǎn)成JSON字符串格式(這樣做是為了適配AndroidiOS可以直接接受JSON對象)然后再通過Toyun這個橋梁去調(diào)用原生應(yīng)用的- (void)share:(NSString *)shareInfo方法這個是有傳參的博敬,參數(shù)為shareInfo友浸。而下面的兩個方法為原生方法調(diào)用后的回調(diào)方法,其中picCallback為獲取圖片成功的回調(diào)方法偏窝,并且傳回拿到的圖片photos收恢;shareCallback為分享成功的回調(diào)方法。

iOS

iOS這邊根據(jù)前端定義的方法名來寫代碼祭往,但是有些時候web前端會讓我們定義伦意,但是我們定義好之后他又要修改,這時候就會很煩啊硼补。所以碰到三端交互的時候最好就是讓web前端去定義方法名驮肉,iOSAndroid根據(jù)web前端定義好的去寫代碼。JavaScriptCoreweb頁面調(diào)用原生應(yīng)用的方法可以用DelegateBlock兩種方法已骇,此文以按Delegate講解离钝。

JavaScriptCore中類及協(xié)議:
  • JSContext:給JavaScript提供運行的上下文環(huán)境
  • JSValue:JavaScriptObjective-C數(shù)據(jù)和方法的橋梁
  • JSManagedValue:管理數(shù)據(jù)和方法的類
  • JSVirtualMachine:處理線程相關(guān)票编,使用較少
  • JSExport:這是一個協(xié)議,如果采用協(xié)議的方法交互卵渴,自己定義的協(xié)議必須遵守此協(xié)議
ViewController中的代碼
#import "ViewController.h"
#import <JavaScriptCore/JavaScriptCore.h>

@protocol JSObjcDelegate <JSExport>

- (void)callCamera;
- (void)share:(NSString *)shareString;

@end

@interface ViewController () <UIWebViewDelegate, JSObjcDelegate>

@property (nonatomic, strong) JSContext *jsContext;
@property (weak, nonatomic) IBOutlet UIWebView *webView;

@end

@implementation ViewController

#pragma mark - Life Circle

- (void)viewDidLoad {
    [super viewDidLoad];
    
    NSURL *url = [[NSBundle mainBundle] URLForResource:@"test" withExtension:@"html"];
    [self.webView loadRequest:[[NSURLRequest alloc] initWithURL:url]];
    
}

#pragma mark - UIWebViewDelegate

- (void)webViewDidFinishLoad:(UIWebView *)webView {
    self.jsContext = [webView valueForKeyPath:@"documentView.webView.mainFrame.javaScriptContext"];
    self.jsContext[@"Toyun"] = self;
    self.jsContext.exceptionHandler = ^(JSContext *context, JSValue *exceptionValue) {
        context.exception = exceptionValue;
        NSLog(@"異常信息:%@", exceptionValue);
    };
}

#pragma mark - JSObjcDelegate

- (void)callCamera {
    NSLog(@"callCamera");
    // 獲取到照片之后在回調(diào)js的方法picCallback把圖片傳出去
    JSValue *picCallback = self.jsContext[@"picCallback"];
    [picCallback callWithArguments:@[@"photos"]];
}

- (void)share:(NSString *)shareString {
    NSLog(@"share:%@", shareString);
    // 分享成功回調(diào)js的方法shareCallback
    JSValue *shareCallback = self.jsContext[@"shareCallback"];
    [shareCallback callWithArguments:nil];
}

@end
ViewController中的代碼解釋

自定義JSObjcDelegate協(xié)議慧域,而且此協(xié)議必須遵守JSExport這個協(xié)議,自定義協(xié)議中的方法就是暴露給web頁面的方法浪读。在webView加載完畢的時候獲取JavaScript運行的上下文環(huán)境吊趾,然后再注入橋梁對象名為Toyun,承載的對象為self即為此控制器瑟啃,控制器遵守此自定義協(xié)議實現(xiàn)協(xié)議中對應(yīng)的方法。在JavaScript調(diào)用完本地應(yīng)用的方法做完相對應(yīng)的事情之后揩尸,又回調(diào)了JavaScript中對應(yīng)的方法蛹屿,從而實現(xiàn)了web頁面本地應(yīng)用之間的通訊。

JavaScriptCore使用注意

JavaScript調(diào)用本地方法是在子線程中執(zhí)行的岩榆,這里要根據(jù)實際情況考慮線程之間的切換错负,而在回調(diào)JavaScript方法的時候最好是在剛開始調(diào)用此方法的線程中去執(zhí)行那段JavaScript方法的代碼,我在實際運用中開始沒注意勇边,就被坑慘了啊犹撒。什么,說的太繞粒褒,看下面的代碼解釋:

//  假設(shè)此方法是在子線程中執(zhí)行的识颊,線程名sub-thread
- (void)callCamera {     
    // 這句假設(shè)要在主線程中執(zhí)行,線程名main-thread
    NSLog(@"callCamera");  

    // 下面這兩句代碼最好還是要在子線程sub-thread中執(zhí)行啊
    JSValue *picCallback = self.jsContext[@"picCallback"];
    [picCallback callWithArguments:@[@"photos"]];
}
運行效果

運行效果如圖3-1所示

圖3-1 運行效果

攔截協(xié)議

攔截協(xié)議這個適合一些比較簡單的一些情況奕坟,不需要引入什么框架祥款,只需要web前端配合一下就好。但是在具體調(diào)用哪一個方法上月杉,以及在傳值的時候可能會有些不方便刃跛,而且調(diào)用完后無法在回調(diào)JavaScript的方法。

web前端

test.html中的代碼
<!DOCTYPE html>
<html>
<head lang="en">
    <meta charset="UTF-8">
</head>
<body>
    <div>
        <input type="button" value="CallCamera" onclick="callCamera()">
    </div>

<script>
    function callCamera() {
        window.location.href = 'toyun://callCamera';
    }
</script>
</body>
</html>
test.html中的代碼解釋

這段代碼相比上面的那段測試代碼是很簡單的苛萎,同樣有一個按鈕桨昙,名字為CallCamera點擊之后調(diào)用自己的callCamera方法,window.location.href這里是改變主窗口的指向從而馬上發(fā)出一個鏈接為toyun://callCamera請求腌歉,而想要傳給原生應(yīng)用的參數(shù)也可已包含到此請求中蛙酪,而在iOS方法中我們要攔截這個請求,根據(jù)請求內(nèi)容去判斷JavaScript想要做的事情究履,從而實現(xiàn)web頁面本地應(yīng)用之間的交互滤否。

iOS

iOS對應(yīng)的代碼
- (BOOL)webView:(UIWebView *)webView shouldStartLoadWithRequest:(NSURLRequest *)request navigationType:(UIWebViewNavigationType)navigationType
{
    NSString *url = request.URL.absoluteString;
    if ([url rangeOfString:@"toyun://"].location != NSNotFound) { 
        // url的協(xié)議頭是toyun
        NSLog(@"callCamera");
        return NO;
    }
    return YES;
}
iOS對應(yīng)的代碼的解釋

webView的代理方法中去攔截自定義的協(xié)議Toyun://如果是此協(xié)議則據(jù)此判斷JavaScript想要做的事情,調(diào)用原生應(yīng)用的方法最仑,這些都是提前約定好的藐俺,同時阻止此鏈接的跳轉(zhuǎn)炊甲。

總結(jié)

隨著手機硬件的配置越來越強大和HTML5的興起,一個App完全可以由web頁面來寫∮郏現(xiàn)在已經(jīng)有部分應(yīng)用這么干了卿啡,我是遇見過的,如古詩文網(wǎng)菱父。盡管比較少但是web頁面本地應(yīng)用的交互不論是iOS還是Android都是會有遇到的颈娜。iOS我還是比較推薦JavaScriptCore,這樣三端可以相對統(tǒng)一起來浙宜,寫的時候都比較簡單官辽。隨著時間的推移iOS8推出的WKWebView會逐漸成為主流,這個的功能更強大。攔截協(xié)議也只能說用到比較簡單的一些情況吧粟瞬,復(fù)雜的情況處理相互之間參數(shù)的傳遞還是比較麻煩的同仆,而且這個不能回調(diào)JavaScript的方法,確實喜歡攔截協(xié)議的同學(xué)可以研究WebViewJavaScriptBridge這個第三方庫裙品。對于Android本人也就是略知皮毛而已俗批,就不班門弄斧了,對于一些Android開發(fā)者來說市怎,可以看地第一段的test.html這個頁面的寫法完全是可以適配Android的岁忘。

更新

關(guān)于使用過程中的坑,出了一片續(xù)区匠,具體參看JavaScript和Objective-C交互的那些事(續(xù))

關(guān)于WKWebView干像,已經(jīng)出了一篇新文章,具體參看WKWebView使用及注意點(keng)

參考

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末辱志,一起剝皮案震驚了整個濱河市蝠筑,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌揩懒,老刑警劉巖什乙,帶你破解...
    沈念sama閱讀 216,402評論 6 499
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異已球,居然都是意外死亡臣镣,警方通過查閱死者的電腦和手機,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,377評論 3 392
  • 文/潘曉璐 我一進店門智亮,熙熙樓的掌柜王于貴愁眉苦臉地迎上來忆某,“玉大人,你說我怎么就攤上這事阔蛉∑妫” “怎么了?”我有些...
    開封第一講書人閱讀 162,483評論 0 353
  • 文/不壞的土叔 我叫張陵,是天一觀的道長聋呢。 經(jīng)常有香客問我苗踪,道長,這世上最難降的妖魔是什么削锰? 我笑而不...
    開封第一講書人閱讀 58,165評論 1 292
  • 正文 為了忘掉前任通铲,我火速辦了婚禮,結(jié)果婚禮上器贩,老公的妹妹穿的比我還像新娘颅夺。我一直安慰自己,他們只是感情好蛹稍,可當(dāng)我...
    茶點故事閱讀 67,176評論 6 388
  • 文/花漫 我一把揭開白布吧黄。 她就那樣靜靜地躺著,像睡著了一般唆姐。 火紅的嫁衣襯著肌膚如雪稚字。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 51,146評論 1 297
  • 那天厦酬,我揣著相機與錄音,去河邊找鬼瘫想。 笑死仗阅,一個胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的国夜。 我是一名探鬼主播减噪,決...
    沈念sama閱讀 40,032評論 3 417
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼车吹!你這毒婦竟也來了筹裕?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 38,896評論 0 274
  • 序言:老撾萬榮一對情侶失蹤窄驹,失蹤者是張志新(化名)和其女友劉穎朝卒,沒想到半個月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體乐埠,經(jīng)...
    沈念sama閱讀 45,311評論 1 310
  • 正文 獨居荒郊野嶺守林人離奇死亡抗斤,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 37,536評論 2 332
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發(fā)現(xiàn)自己被綠了丈咐。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片瑞眼。...
    茶點故事閱讀 39,696評論 1 348
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖棵逊,靈堂內(nèi)的尸體忽然破棺而出伤疙,到底是詐尸還是另有隱情,我是刑警寧澤辆影,帶...
    沈念sama閱讀 35,413評論 5 343
  • 正文 年R本政府宣布徒像,位于F島的核電站黍特,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏厨姚。R本人自食惡果不足惜衅澈,卻給世界環(huán)境...
    茶點故事閱讀 41,008評論 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望谬墙。 院中可真熱鬧今布,春花似錦、人聲如沸拭抬。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,659評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽造虎。三九已至傅蹂,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間算凿,已是汗流浹背份蝴。 一陣腳步聲響...
    開封第一講書人閱讀 32,815評論 1 269
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留氓轰,地道東北人婚夫。 一個月前我還...
    沈念sama閱讀 47,698評論 2 368
  • 正文 我出身青樓,卻偏偏與公主長得像署鸡,于是被迫代替她去往敵國和親案糙。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 44,592評論 2 353

推薦閱讀更多精彩內(nèi)容