iOS OC与JavaScript的交互

iOS OC与JavaScript的交互

概念了解

JavaScriptCore

javaScriptCore是iOS7后推出的框架,是封装了JavaScript和Objective-C桥接的Objective-C API,我们只需要只要用很少的代码,就可以做到JavaScript调用Objective-C,或者Objective-C调用JavaScript。

JavaScriptCore中类及协议

  • JSManagedValue:管理数据和方法的类
  • JSContent:JS执行的环境
  • JSValue:JS和OC数据和方法的桥梁
  • JSVirtualMachine:处理线程相关,使用较少
  • JSExport:这是一个协议,如果JS对象想直接调用OC对象里面的方法和属性,那么这个OC对象只要实现这个JSExport协议就可以了。

代码示例

我们先用终端创建个html文件拖入工程

test.html中代码如下

<!doctype html>
<html>
<head>
    <meta charset="UTF-8">
        
        <meta name="viewport" content="width=device-width, initial-scale=1.0, minimum-scale=1.0, maximum-scale=1.0, user-scalable=no, target-densitydpi=device-dpi"/>
        
        <title>JSCallOC</title>
        
        <style>
            *
            {
                //-webkit-tap-highlight-color: rgba(0,0,0,0);
                text-decoration: none;
            }
        
        html,body
        {
            -webkit-touch-callout: none;                /* prevent callout to copy image, etc when tap to hold */
            -webkit-text-size-adjust: none;             /* prevent webkit from resizing text to fit */
            -webkit-user-select: none;                  /* prevent copy paste, to allow, change 'none' to 'text' */
        }
        
        #div-a
        {
            background:#FBA;
            color:#FFF;
            
            border-radius: 25px 5px;
        }
        
        
            </style>
        
        <script type="text/javascript">
            
            function showResult(resultNumber)
            {
                //alert(resultNumber);
                document.getElementById("result").innerText = resultNumber;

            }
        
            function picCallBack(image) {
                alert(image);
            }
        
            </script>
        
</head>

<body style="background:#CDE; color:#FFF">
    
    <div>
        <font size="3" color="black">输入一个整数:</font>
        <textarea  id="input" style="font-size:10pt;color:black;"></textarea>
    </div>
    <br/>
    
    <div>
        <font size="3" color="black">结果: <b id="result"> </b> </font>
    </div>
    <br/>
    
    <div id="div-a">
        <center>
            
            <br/>
            <input type="button" value="计算阶乘" onclick="native.calculateForJS(input.value);" />
            <br/>
            <br/>
            
            
            <input type="button" value="测试log" onclick="log('测试');" />
            <br/>
            <br/>
            
            <input type="button" value="oc原生Alert" onclick="alert('alert');" />
            <br/>
            <br/>
            
            <input type="button" value="addSubView" onclick="addSubView('view');" />
            <br/>
            <br/>
            
            <input type="button" value="removeSubView" onclick="removeSubView('view');" />
            <br/>
            <br/>
            
            <input type="button" value="多参数调用" onclick="mutiParams('参数1','参数2','参数3');" />
            <br/>
            <br/>
            
            <input type="button" value="获取照片" onclick="native.callCamera()" />
            <br/>
            <br/>
            <a id="push" href="#" onclick="native.pushViewControllerTitle('SecondViewController','secondPushedFromJS');">
                push to second ViewController
            </a>
            
            <br/>
            <br/>
            
        </center>
    </div>
    
    
    
</body>

</html>
运行效果图

整个页面均为HTML实现,功能为:

1 计算阶乘:输入框输入数字后调用OC中相关方法进行计算,将计算结果显示在HTML页面上。

2 测试log:点击后,在控制台打印测试数据。

3 OC原生Alert:点击后,弹出OC的提示框。

4 addSubView:点击后,在OC中添加一个View.

5 removeSubView: 点击后,移除4中添加的View。

6 多函数调用: 获取HTML中的多个参数

7 获取照片:访问手机照片,并将选中照片显示在HTML页面上

8 push to Second View Controller:跳转到下一个页面。

总结:以上功能都是OC中获取HTML按钮中的相关点击事件,然后在OC中执行相关代码。

ViewController.m中代码如下

#import "OneViewController.h"
#import <JavaScriptCore/JavaScriptCore.h>
#import "SecondViewController.h"

@protocol TestJSExport <JSExport>
/*
 OC的函数命名和JS函数命名规则不同 我们可以通过JSExportAs这个宏优化JS中调用的名称
 这个宏只对有参数的selector起作用
 handleFactorialCalculateWithNumber:(NSNumber *)number作为 js方法:calculateForJS的别名*/
JSExportAs
(calculateForJS, - (void)handleFactorialCalculateWithNumber:(NSNumber *)number);
//- (void)calculateForJS:(NSNumber *)number;
//js方法
- (void)pushViewController:(NSString *)view title:(NSString *)title;
- (void)callCamera;

@end

@interface OneViewController ()<UIWebViewDelegate, TestJSExport>

@property (nonatomic, strong) UIWebView *webView;
@property (nonatomic, strong) JSContext *context;//给JavaScript提供运行的上下文环境
@property (nonatomic, strong) UIView *addView;

@end

@implementation OneViewController

- (void)viewDidLoad {
    [super viewDidLoad];
    
    [self.view addSubview:self.webView];
    
    NSString *path = [[[NSBundle mainBundle] bundlePath] stringByAppendingPathComponent:@"test.html"];
    NSString *htmlString = [NSString stringWithContentsOfFile:path encoding:NSUTF8StringEncoding error:nil];
    [_webView loadHTMLString:htmlString baseURL:nil];
    
}

- (UIWebView *)webView {
    if (_webView == nil) {
        _webView = [[UIWebView alloc] initWithFrame:self.view.bounds];
        _webView.delegate = self;
    }
    return _webView;
}

- (UIView *)addView {
    if (_addView == nil) {
        _addView =[[UIView alloc] initWithFrame:CGRectMake(10, 550, 200, 100)];
        _addView.backgroundColor = [UIColor cyanColor];
    }
    return _addView;
}

#pragma mark UIWebViewDelegate
- (void)webViewDidFinishLoad:(UIWebView *)webView {
    //将 html的title 设置为controller的title
    self.title = [webView stringByEvaluatingJavaScriptFromString:@"document.title"];
    //获取当前页面的url
    NSString *url = [webView stringByEvaluatingJavaScriptFromString:@"document.location.href"];
    //这个好像是私有属性 审核时可能被苹果拒绝
    self.context = [webView valueForKeyPath:@"documentView.webView.mainFrame.javaScriptContext"];
    //打印异常,由于JS的异常信息是不会在OC中被直接打印的,所以我们在这里添加打印异常信息
    self.context.exceptionHandler = ^(JSContext *context, JSValue *exceptionValue) {
        context.exception = exceptionValue;
        NSLog(@"exceptionValue --- %@",exceptionValue);
    };
    //以 JSExport 协议关联 native方法
    self.context[@"native"] = self;
    //以 block 形式关联 JavaScript function
    self.context[@"log"] = ^(NSString *str) {
        NSLog(@"%@",str);
    };
    //以 block 形式关联 JavaScript function
    self.context[@"alert"] = ^(NSString *str) {
        dispatch_async(dispatch_get_main_queue(), ^{
            UIAlertView *alter = [[UIAlertView alloc] initWithTitle:@"msg from js" message:str delegate:nil cancelButtonTitle:@"ok" otherButtonTitles:nil, nil];
            [alter show];
        });
    };
    //弱引用 避免循环引用
    __block typeof(self) weakSelf = self;
    self.context[@"addSubView"] = ^(NSString *viewName) {
        [weakSelf.view addSubview:weakSelf.addView];
    };
    
    self.context[@"removeSubView"] = ^(NSString *viewName) {
        [weakSelf.addView removeFromSuperview];
    };
    //多参数
    self.context[@"mutiParams"] = ^(NSString *a, NSString *b, NSString *c) {
        NSLog(@"%@ %@ %@",a,b,c);
    };
}

#pragma mark - JSExport Methods
- (void)handleFactorialCalculateWithNumber:(NSNumber *)number{
    NSLog(@"%@", number);
    NSNumber *result = [self calculateFactorialOfNumber:number];
    NSLog(@"%@", result);
    [self.context[@"showResult"] callWithArguments:@[result]];
}

- (void)pushViewController:(NSString *)view title:(NSString *)title{
    Class second = NSClassFromString(view);
    id secondVC = [[second alloc]init];
    ((UIViewController*)secondVC).title = title;
    [self.navigationController pushViewController:secondVC animated:YES];
}

//  假设此方法是在子线程中执行的,线程名sub-thread
- (void)callCamera {
    // 这句假设要在主线程中执行,线程名main-thread
    NSLog(@"callCamera");
    
    // 下面这两句代码最好还是要在子线程sub-thread中执行啊
    JSValue *picCallback = self.context[@"picCallBack"];
    [picCallback callWithArguments:@[@"photos"]];
}

- (void)calculateForJS:(NSNumber *)number {
    NSLog(@"点击了计算阶乘");
    
    JSValue *showResult = self.context[@"showResult"];
    [showResult callWithArguments:@[@"计算阶乘"]];
    
}

#pragma mark - Factorial Method
- (NSNumber *)calculateFactorialOfNumber:(NSNumber *)number{
    NSInteger i = [number integerValue];
    if (i < 0){
        return [NSNumber numberWithInteger:0];
    }
    if (i == 0){
        return [NSNumber numberWithInteger:1];
    }
    NSInteger r = (i * [(NSNumber *)[self calculateFactorialOfNumber:[NSNumber numberWithInteger:(i - 1)]] integerValue]);
    
    return [NSNumber numberWithInteger:r];
}

- (void)viewWillDisappear:(BOOL)animated {
    [super viewWillDisappear:animated];
    self.context[@"native"] = nil;
}


@end

获取HTML中的点击事件

在HTML中,为一个元素添加点击事件的两种方法

第一种

<input type="button" value="计算阶乘" onclick="native.calculateForJS(input.value);" />

在JS交互中,很多事情都是在webView的delegate方法中完成的,通过JSContent创建一个使用JS的环境,所以这里,我们先将self.content在这里面初始化;

#pragma mark UIWebViewDelegate
- (void)webViewDidFinishLoad:(UIWebView *)webView {
    //这个好像是私有属性 审核时可能被苹果拒绝
    self.context = [webView valueForKeyPath:@"documentView.webView.mainFrame.javaScriptContext"];
    //打印异常,由于JS的异常信息是不会在OC中被直接打印的,所以我们在这里添加打印异常信息
    self.context.exceptionHandler = ^(JSContext *context, JSValue *exceptionValue) {
        context.exception = exceptionValue;
        NSLog(@"exceptionValue --- %@",exceptionValue);
    };
    //以JSExport 协议关联 native 的方法
    self.context[@"native"] = self;
}

我们需要声明一个集成JSExport协议,协议中声明JS使用的OC方法

@protocol TestJSExport <JSExport>
/*
 OC的函数命名和JS函数命名规则不同 我们可以通过JSExportAs这个宏优化JS中调用的名称
 这个宏只对有参数的selector起作用
 handleFactorialCalculateWithNumber:(NSNumber *)number作为 js方法:calculateForJS的别名*/
JSExportAs
(calculateForJS, - (void)handleFactorialCalculateWithNumber:(NSNumber *)number);
@end

当然你也可以按下面的写法

@protocol TestJSExport <JSExport>

- (void)calculateForJS:(NSNumber *)number;

@end

第二种

<input type="button" value="oc原生Alert" onclick="alert('alert');" />

这种我们需要使用block的形式关联JavaScript function

self.context[@"alert"] = ^(NSString *str) {

};

对HTML中的事件进行处理

第一种 协议形式

我们协议中制定的方法名一定要和HTML中的方法名相同。
当我们协议需要使用JS中的方法时,用下面的代码进行调用:

HTML中的方法

function showResult(resultNumber)
{
document.getElementById("result").innerText = resultNumber;

}

OC调用

JSValue *showResult = self.context[@"showResult"];
[showResult callWithArguments:@[@"计算阶乘"]];

第二种 Block形式

注意避免循环引用,同时刷新UI的工作应该放到主线程

self.context[@"alert"] = ^(NSString *str) {
    dispatch_async(dispatch_get_main_queue(), ^{
        UIAlertView *alter = [[UIAlertView alloc] initWithTitle:@"msg from js" message:str delegate:nil cancelButtonTitle:@"ok" otherButtonTitles:nil, nil];
        [alter show];
    });
};

使用注意

OC调用JavaScript是同步,JavaScript调用OC是异步

JavaScript调用本地方法是在子线程中执行的,这里要根据实际情况考虑线程之间的切换,而在回调JavaScript方法的时候最好是在刚开始调用此方法的线程中去执行那段JavaScript方法的代码,看下面的代码解释:

// 假设此方法是在子线程中执行的,线程名sub-thread
- (void)callCamera {
    // 这句假设要在主线程中执行,线程名main-thread
    NSLog(@"callCamera");
    
    // 下面这两句代码最好还是要在子线程sub-thread中执行啊
    JSValue *picCallback = self.context[@"picCallBack"];
    [picCallback callWithArguments:@[@"photos"]];
}

本文demo: 点我下载

内存管理陷阱

Objective-C的内存管理机制是引用计数,JavaScript的内存管理机制是垃圾回收。在大部分情况下,JavaScriptCore能做到在这两种内存管理机制之间无缝无错转换,但也有少数情况需要特别注意。

在block内捕获JSContext

Block会为默认为所有被它捕获的对象创建一个强引用。JSContext为它管理的所有JSValue也都拥有一个强引用。并且,JSValue会为它保存的值和它所在的Context都维持一个强引用。这样JSContext和JSValue看上去是循环引用的,然而并不会,垃圾回收机制会打破这个循环引用。
看下面的例子:

self.context[@"getVersion"] = ^{
    NSString *versionString = [[NSBundle mainBundle] objectForInfoDictionaryKey:@"CFBundleShortVersionString"];

    versionString = [@"version " stringByAppendingString:versionString];

    JSContext *context = [JSContext currentContext]; // 这里不要用self.context
    JSValue *version = [JSValue valueWithObject:versionString inContext:context];

    return version;
};

使用[JSContext currentContext]而不是self.context来在block中使用JSContext,来防止循环引用。

JSManagedValue

当把一个JavaScript值保存到一个本地实例变量上时,需要尤其注意内存管理陷阱。 用实例变量保存一个JSValue非常容易引起循环引用。

看以下下例子,自定义一个UIAlertView,当点击按钮时调用一个JavaScript函数:

#import <UIKit/UIKit.h>
#import <JavaScriptCore/JavaScriptCore.h>

@interface MyAlertView : UIAlertView

- (id)initWithTitle:(NSString *)title
            message:(NSString *)message
            success:(JSValue *)successHandler
            failure:(JSValue *)failureHandler
            context:(JSContext *)context;

@end

按照一般自定义AlertView的实现方法,MyAlertView需要持有successHandler,failureHandler这两个JSValue对象

向JavaScript环境注入一个function

self.context[@"presentNativeAlert"] = ^(NSString *title,
                                        NSString *message,
                                        JSValue *success,
                                        JSValue *failure) {
   JSContext *context = [JSContext currentContext];
   MyAlertView *alertView = [[MyAlertView alloc] initWithTitle:title 
                                                       message:message
                                                       success:success
                                                       failure:failure
                                                       context:context];
   [alertView show];
};

因为JavaScript环境中都是“强引用”(相对Objective-C的概念来说)的,这时JSContext强引用了一个presentNativeAlert函数,这个函数中又强引用了MyAlertView 等于说JSContext强引用了MyAlertView,而MyAlertView为了持有两个回调强引用了successHandler和failureHandler这两个JSValue,这样MyAlertView和JavaScript环境互相引用了。

所以苹果提供了一个JSMagagedValue类来解决这个问题。

看MyAlertView.m的正确实现:

#import "MyAlertView.h"

@interface XorkAlertView() <UIAlertViewDelegate>
@property (strong, nonatomic) JSContext *ctxt;
@property (strong, nonatomic) JSMagagedValue *successHandler;
@property (strong, nonatomic) JSMagagedValue *failureHandler;
@end

@implementation MyAlertView

- (id)initWithTitle:(NSString *)title
            message:(NSString *)message
            success:(JSValue *)successHandler
            failure:(JSValue *)failureHandler
            context:(JSContext *)context {

    self = [super initWithTitle:title
                    message:message
                   delegate:self
          cancelButtonTitle:@"No"
          otherButtonTitles:@"Yes", nil];

    if (self) {
        _ctxt = context;

        _successHandler = [JSManagedValue managedValueWithValue:successHandler];
        // A JSManagedValue by itself is a weak reference. You convert it into a conditionally retained
        // reference, by inserting it to the JSVirtualMachine using addManagedReference:withOwner:
        [context.virtualMachine addManagedReference:_successHandler withOwner:self];

        _failureHandler = [JSManagedValue managedValueWithValue:failureHandler];
        [context.virtualMachine addManagedReference:_failureHandler withOwner:self];
    }
    return self;
}

- (void)alertView:(UIAlertView *)alertView clickedButtonAtIndex:(NSInteger)buttonIndex {
    if (buttonIndex == self.cancelButtonIndex) {
        JSValue *function = [self.failureHandler value];
        [function callWithArguments:@[]];
    } else {
        JSValue *function = [self.successHandler value];
        [function callWithArguments:@[]];
    }

    [self.ctxt.virtualMachine removeManagedReference:_failureHandler withOwner:self];
    [self.ctxt.virtualMachine removeManagedReference:_successHandler withOwner:self];
}    

@end

分析上面例子,从外部传入的JSValue对象在类内部使用JSManagedValue来保存。

JSManagedValue本身是一个弱引用对象,需要调用JSVirtualMachine的addManagedReference:withOwner:把它添加到JSVirtualMachine对象中,确保使用过程中JSValue不会被释放

当用户点击AlertView上的按钮时,根据用户点击哪一个按钮,来执行对应的处理函数,这时AlertView也随即被销毁。 这时需要手动调用removeManagedReference:withOwner:来移除JSManagedValue。

参考文章

http://www.jianshu.com/p/cdaf9bc3d65d
https://hjgitbook.gitbooks.io/ios/content/04-technical-research/04-javascriptcore-note.html
http://my.oschina.net/whforever/blog/669813
http://www.jianshu.com/p/f896d73c670a

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,686评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,668评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,160评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,736评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,847评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,043评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,129评论 3 410
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,872评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,318评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,645评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,777评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,470评论 4 333
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,126评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,861评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,095评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,589评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,687评论 2 351

推荐阅读更多精彩内容