iOS OC與JS 相互調用

OC調用JS

  • 首先我們創建一個WebView,加載一個本地的HTML文件
//加載web  本地的一個html文件
    NSURL * url = [[NSBundle mainBundle] URLForResource:@"index" withExtension:@"html"];
    NSURLRequest * request = [NSURLRequest requestWithURL:url];
    [self.webView loadRequest:request];
  • webview加載完成時-(void)webViewDidFinishLoad:(UIWebView *)webView,我們調用html文件中的頭部標題設置為導航控制器的title
    NSString * title = [webView stringByEvaluatingJavaScriptFromString:@"document.title"];
    self.title = title;
  • - (nullable NSString *)stringByEvaluatingJavaScriptFromString:(NSString *)script;這個方法就是我們在OC中調用JS的核心方法,我們只需要把寫好的js以字符串的形式傳入webview就可以執行document.title這段語法,這段語法返回的標題就會通過這個方法的返回值返回給我們
  • 一個簡單的小例子,html文件中有一個testjs方法,同樣在webview加載完成時我們以同樣的方法將test()傳入給webviewwebview就會執行這個test方法
  • html中的代碼
<script>
        function test(){
        
                alert("ceshi");
          }
 </script>
  • OC 中的代碼
   [webView stringByEvaluatingJavaScriptFromString:@"test();"];
  • 效果就是在加載完成時彈出一個我們前端開發最常見的alert彈框

JS調用OC

  • 在JS中調用OC中的方法比較蛋疼,在webView中我們可以通過代理方法捕捉網頁的去向,也就是a標簽的href屬性,或者網頁的location.href.
  • 我們自定義一個我們自己的協議,例如location.href = 'heihei://ceshi';當我們點擊js中的一個按鈕時onclick="ceshi();"執行location.href的跳轉,這樣我們在OC中就可以捕捉到request.URL.absoluteString;這個跳轉的url
  • 當我們捕捉到這個url只需要判斷下是否是我們自定的協議,如果是我們這個heihei://協議,就根據url執行OC中相應的方法,即我們獲得到了heihei://ceshi,我們就應該執行OC中名字為ceshi的方法,我們可以根據字符串截取到ceshi,然后利用方法響應器- (id)performSelector:(SEL)aSelector;來調用-(void)ceshi{}
  • 在來說一下傳參數的調用,在OC中的方法名帶有參數是以的方式拼接的,然而在js:是無法存在于方法名中的,在這里,我們用_來代替,并且通過?拼接參數,參數與參數之間以&隔開,例如一個參數就是這樣heihei://ceshi_?100,兩個參數就是這樣heihei://ceshi_number2_?100&200,依次類推。
  • 這樣我們開始在OC中對捕捉到url進行處理,首先截掉前面我們自定義的協議,然后以?分割,第一部分就是ceshi_number2_
  • 然后我們用:替換_就變成了ceshi:number2:這就獲得到了OC中- (void)ceshi:(NSString *)number1 number2:(NSString *)number2{的方法名,仔細比對下哦,看看有沒有問題哦!
  • 下一步我們處理參數,剛才以分割的字符串的后半部分為100&200這就是我們需要的參數,我們將這個字符串以&分割成數組,就獲取到了我們的參數數組
  • 最后通過同樣的方式,方法選擇器通過方法名來調用方法
-(id)performSelector:(SEL)aSelector withObject:(id)object;
-(id)performSelector:(SEL)aSelector withObject:(id)object1 withObject:(id)object2;
  • 但是我們找遍了蘋果performSelector方法也就只找到了上面三種,那么現在問題來了,如果傳三個參數,或者四個參數,我們應該如何處理呢。顯然,這樣是有局限性的。
  • 所以我們要模仿蘋果自己寫一個performSelector方法,寫一個比較通用的,可以多傳參數的方法,這里我們就用到了NSInvocation這個類,這是一個利用一個NSInvocation對象包裝一次方法調用(方法調用者、方法名、方法參數、方法返回值)
  • 我們自己封裝了一個- (id)performSelect:(SEL)selector withObjects:(NSArray *)objects通過傳參數的方法選擇器和參數數組調用方法的方式,代碼如下
-(id)performSelect:(SEL)selector withObjects:(NSArray *)objects{
    //方法簽名(方法描述)
    NSMethodSignature * signature = [self methodSignatureForSelector:selector];
    if (signature == nil) {
        //找不到該方法拋出異常
        @throw [NSException exceptionWithName:@"牛逼的錯誤" reason:@"方法找不到" userInfo:nil];
    }
    // NSInvocation : 利用一個NSInvocation對象包裝一次方法調用(方法調用者、方法名、方法參數、方法返回值)
    NSInvocation * invocation = [NSInvocation invocationWithMethodSignature:signature];
    //設置響應者
    invocation.target = self;
    //設置方法選擇器
    invocation.selector = selector;
    //通過循環設置參數
    NSInteger paramsCount = signature.numberOfArguments;
    paramsCount = MIN(paramsCount, objects.count);
    for (NSInteger i = 0; i<paramsCount; i++) {
        id object = objects[i];
        if ([object isKindOfClass:[NSNull class]]) {
            continue;
        }
        [invocation setArgument:&object atIndex:i+2];
    }
    //調用方法
    [invocation invoke];
    //獲取返回值
    id returnValue = nil;
    //當返回值不是空的時候
    if (signature.methodReturnLength) {
        [invocation getReturnValue:&returnValue];
    }
    return returnValue;
}
  • 我們捕捉href跳轉時webview的攔截代理方法就可以寫成下面的樣子
//捕獲location.href的跳轉信息
-(BOOL)webView:(UIWebView *)webView shouldStartLoadWithRequest:(NSURLRequest *)request navigationType:(UIWebViewNavigationType)navigationType{

    //獲取跳轉的URL
    NSString * url = request.URL.absoluteString;
    //自定義的協議
    NSString * scheme = @"heihei://";
    //如果url中是我們自定義的協議,實現OC中的方法
    if ([url containsString:scheme]) {
        //截取自定義協議后面的字符串
        NSString * path = [url substringFromIndex:scheme.length];
        NSString * method = nil;
        NSArray * params = nil;
        //字符串是否包含?,判斷是否有參數
        if ([path containsString:@"?"]) {
            //有參數
            //通過?分割,
            NSArray * info = [path componentsSeparatedByString:@"?"];
            //數組第一個元素就是方法名,將js中的_替換為:轉換為OC中的方法名
            method = [[info firstObject] stringByReplacingOccurrencesOfString:@"_" withString:@":"];
            //數組第二個元素就為參數字符串
            NSString * param = [info lastObject];
            //將參數以&分割變為參數數組
            params = [param componentsSeparatedByString:@"&"];
        }else{
            //無參數
            //方法名
            method = path;
            //參數數組
            params = @[];
        }
        NSLog(@"method:%@------params:%@",method,params);
        //通過方法簽名來調用方法,因為參數的個數未知,所以我們封裝一個比較通用的方法
        [self performSelect:NSSelectorFromString(method) withObjects:params];

        return NO;
    }
    return YES;
}
  • html中按照我們的規則進行跳轉
<script>
            function ceshi()
            {
                location.href = 'heihei://ceshi';
            }
            function ceshi1()
            {
                location.href = 'heihei://ceshi_?100';
            }
            function ceshi2()
            {
                location.href = 'heihei://ceshi_number2_?100&200';
            }
            function ceshi3()
            {
                location.href = 'heihei://ceshi_number2_number3_?100&200&300';
            }
        </script>

Demo全部代碼

  • index.html
<html>
    <!-- 網頁的描述信息 -->
    <head>
        <meta charset="UTF-8">
        <title>HeiheiJSWithOC</title>
            
        <script>
            function ceshi()
            {
                //沒有參數時
                location.href = 'heihei://ceshi';
            }
            function ceshi1()
            {
                //一個參數時
                location.href = 'heihei://ceshi_?100';
            }
            function ceshi2()
            {
                //二個參數時
                location.href = 'heihei://ceshi_number2_?100&200';
            }
            function ceshi3()
            {
                //三個參數時
                location.href = 'heihei://ceshi_number2_number3_?100&200&300';
            }
            function test(){
        
                alert("ceshi");
            }
        </script>
    </head>
    <!-- 網頁的具體內容 -->
    <body>
        <div>電話:10086</div>
        <br>
        <button style="color:white ;background-color: red; width:100px; height:30px; margin-top:5px" onclick="ceshi();">測試無參數</button>
        <br>
        <button style="color:white ;background-color: red; width:100px; height:30px; margin-top:5px" onclick="ceshi1();">測試1個參數</button>
        <br>
        <button style="color:white ;background-color: red; width:100px; height:30px; margin-top:5px" onclick="ceshi2();">測試2個參數</button>
        <br>
        <button style="color:white ;background-color: red; width:100px; height:30px; margin-top:5px" onclick="ceshi3();">測試3個參數</button>
        <br>
        <a >百度</a>
    </body>
</html>
  • OC中ViewController.m的代碼
//
//  ViewController.m
//  OCWithJS
//
//  Created by zzh on 2017/2/17.
//  Copyright ? 2017年 zzh. All rights reserved.
//
#import "ViewController.h"
@interface ViewController ()<UIWebViewDelegate>
@property (weak, nonatomic) IBOutlet UIWebView *webView;
@end
@implementation ViewController
-(void)viewDidLoad {
    [super viewDidLoad];
    //加載web  本地的一個html文件
    NSURL * url = [[NSBundle mainBundle] URLForResource:@"index" withExtension:@"html"];
    NSURLRequest * request = [NSURLRequest requestWithURL:url];
    [self.webView loadRequest:request];
}
#pragma mark - 不同參數的測試方法
-(void)ceshi{    
    NSLog(@"%s----無參數", __func__);    
}
-(void)ceshi:(NSString *)number1{   
    NSLog(@"%s----%@---", __func__,number1);    
}
-(void)ceshi:(NSString *)number1 number2:(NSString *)number2{    
    NSLog(@"%s----%@---%@--", __func__,number1,number2);   
}
-(void)ceshi:(NSString *)number1 number2:(NSString *)number2 number3:(NSString *)number3{   
    NSLog(@"%s----%@---%@---%@---", __func__,number1,number2,number3);    
}
#pragma mark - UIWebViewDelegate
//捕獲location.href的跳轉信息
-(BOOL)webView:(UIWebView *)webView shouldStartLoadWithRequest:(NSURLRequest *)request navigationType:(UIWebViewNavigationType)navigationType{

    //獲取跳轉的URL
    NSString * url = request.URL.absoluteString;
    //自定義的協議
    NSString * scheme = @"heihei://";
    //如果url中是我們自定義的協議,實現OC中的方法
    if ([url containsString:scheme]) {
        //截取自定義協議后面的字符串
        NSString * path = [url substringFromIndex:scheme.length];
        NSString * method = nil;
        NSArray * params = nil;
        //字符串是否包含?,判斷是否有參數
        if ([path containsString:@"?"]) {
            //有參數
            //通過?分割,
            NSArray * info = [path componentsSeparatedByString:@"?"];
            //數組第一個元素就是方法名,將js中的_替換為:轉換為OC中的方法名
            method = [[info firstObject] stringByReplacingOccurrencesOfString:@"_" withString:@":"];
            //數組第二個元素就為參數字符串
            NSString * param = [info lastObject];
            //將參數以&分割變為參數數組
            params = [param componentsSeparatedByString:@"&"];
        }else{
            //無參數
            //方法名
            method = path;
            //參數數組
            params = @[];
        }
        NSLog(@"method:%@------params:%@",method,params);
        //通過方法簽名來調用方法,因為參數的個數未知,所以我們封裝一個比較通用的方法
        [self performSelect:NSSelectorFromString(method) withObjects:params];

        return NO;
    }
    return YES;
}
-(void)webViewDidFinishLoad:(UIWebView *)webView{
    //oc調用js
    NSString * title = [webView stringByEvaluatingJavaScriptFromString:@"document.title"];
    self.title = title;
   [webView stringByEvaluatingJavaScriptFromString:@"test();"];
}
#pragma mark - 根據方法簽名調用方法
-(id)performSelect:(SEL)selector withObjects:(NSArray *)objects{
    //方法簽名(方法描述)
    NSMethodSignature * signature = [self methodSignatureForSelector:selector];
    if (signature == nil) {
        //找不到該方法拋出異常
        @throw [NSException exceptionWithName:@"牛逼的錯誤" reason:@"方法找不到" userInfo:nil];
    }
    // NSInvocation : 利用一個NSInvocation對象包裝一次方法調用(方法調用者、方法名、方法參數、方法返回值)
    NSInvocation * invocation = [NSInvocation invocationWithMethodSignature:signature];
    //設置響應者
    invocation.target = self;
    //設置方法選擇器
    invocation.selector = selector;
    //通過循環設置參數
    NSInteger paramsCount = signature.numberOfArguments;
    paramsCount = MIN(paramsCount, objects.count);
    for (NSInteger i = 0; i<paramsCount; i++) {
        id object = objects[i];
        if ([object isKindOfClass:[NSNull class]]) {
            continue;
        }
        [invocation setArgument:&object atIndex:i+2];
    }
    //調用方法
    [invocation invoke];
    //獲取返回值
    id returnValue = nil;
    //當返回值不是空的時候
    if (signature.methodReturnLength) {
        [invocation getReturnValue:&returnValue];
    }
    return returnValue;
}
@end
最后編輯于
?著作權歸作者所有,轉載或內容合作請聯系作者
平臺聲明:文章內容(如有圖片或視頻亦包括在內)由作者上傳并發布,文章內容僅代表作者本人觀點,簡書系信息發布平臺,僅提供信息存儲服務。
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市,隨后出現的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 229,117評論 6 537
  • 序言:濱河連續發生了三起死亡事件,死亡現場離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機,發現死者居然都...
    沈念sama閱讀 98,860評論 3 423
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人,你說我怎么就攤上這事。” “怎么了?”我有些...
    開封第一講書人閱讀 177,128評論 0 381
  • 文/不壞的土叔 我叫張陵,是天一觀的道長。 經常有香客問我,道長,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 63,291評論 1 315
  • 正文 為了忘掉前任,我火速辦了婚禮,結果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當我...
    茶點故事閱讀 72,025評論 6 410
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發上,一...
    開封第一講書人閱讀 55,421評論 1 324
  • 那天,我揣著相機與錄音,去河邊找鬼。 笑死,一個胖子當著我的面吹牛,可吹牛的內容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 43,477評論 3 444
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了?” 一聲冷哼從身側響起,我...
    開封第一講書人閱讀 42,642評論 0 289
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后,有當地人在樹林里發現了一具尸體,經...
    沈念sama閱讀 49,177評論 1 335
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內容為張勛視角 年9月15日...
    茶點故事閱讀 40,970評論 3 356
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發現自己被綠了。 大學時的朋友給我發了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 43,157評論 1 371
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 38,717評論 5 362
  • 正文 年R本政府宣布,位于F島的核電站,受9級特大地震影響,放射性物質發生泄漏。R本人自食惡果不足惜,卻給世界環境...
    茶點故事閱讀 44,410評論 3 347
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 34,821評論 0 28
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春,著一層夾襖步出監牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 36,053評論 1 289
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個月前我還...
    沈念sama閱讀 51,896評論 3 395
  • 正文 我出身青樓,卻偏偏與公主長得像,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 48,157評論 2 375

推薦閱讀更多精彩內容