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中代码如下




    
        
        
        
        JSCallOC
        
        
        
        
        



    
    
输入一个整数:

结果:

iOS OC与JavaScript的交互_第1张图片
运行效果图

整个页面均为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 
#import "SecondViewController.h"

@protocol TestJSExport 
/*
 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 ()

@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中,为一个元素添加点击事件的两种方法

第一种


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

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

@protocol TestJSExport 

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

@end

第二种


这种我们需要使用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 
#import 

@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() 
@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

你可能感兴趣的:(iOS OC与JavaScript的交互)