热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

JS和OC相互调用

1、现状:人人都是产品经理聊聊WebApp、HybridApp与NativeApp的设计差异标点符(钱魏Way)NativeApp、WebApp还是HybridApp

1、现状:

人人都是产品经理 聊聊Web App、Hybrid App与Native App的设计差异

标点符(钱魏 Way) Native App、Web App 还是Hybrid App?

image

image

1)Native APP:Native Code编程,代码编译之后以2进制或者字节码的形式运行在OS上,直接调用OS的Device API;

2)Web APP,以HTML+JS+CSS等WEB技术编程,代码运行在浏览器中,通过浏览器来调用Device API(取决于HTML5未来的支持能力):

3)Hybrid APP,部分代码以WEB技术编程,部分代码由某些Native Container承担(例如PhonGAP(Cordova)插件,BAE插件),其目的是在HTML5尚未完全支持Device API和Network API的目前阶段,承担这部分职责。(我们当前所使用的)

2、Hybrid APP问题:

处理原生代码(OC)和HTML5的交互(JS)问题

3、JS调用OC代码:

3.1 实现 UIWebView 的代理,然后根据NSURLRequest的URL进行不同处理,JS中的将要传递的数据作为URL 重定向 :
//webView的代理相应重定向
- (BOOL)webView:(UIWebView *)webView shouldStartLoadWithRequest:(NSURLRequest *)request navigationType:(UIWebViewNavigationType)navigationType{
NSString *requestString = [[[request URL] absoluteString] stringByReplacingPercentEscapesUsingEncoding:NSUTF8StringEncoding];
NSLog(@"should-------");
if ([requestString hasPrefix:@"fdd://"]) {
//根据自己定义的规则,通过字符串的值,调用OC的方法。这里就输出一下字符串了。
NSLog(@"===%@",requestString); //可能包含商品的ID,APP拿到这个ID,使用OC代码执行相关操作
return NO; //YES 表示正常加载网页 返回NO 将停止网页加载
}

return YES;
}

优点:勉强实现JS调用OC代码,容易理解 =_=

缺点:硬编码,扩展性差

3.2 对3.1的一些扩展:

zttjhm的专栏 UIWebView中Html中用JS调用OC方法及OC执行JS代码

HTML代码:












alert提示 //href 要跳转的链接




actionsheet提示


iOS代码:

#pragma clang diagnostic push
#pragma clang diagnostic ignored "-Warc-performSelector-leaks"
- (BOOL)webView:(UIWebView *)webView shouldStartLoadWithRequest:(NSURLRequest *)request navigationType:(UIWebViewNavigationType)navigationType{

NSString *urlstr = request.URL.absoluteString;
NSRange range = [urlstr rangeOfString:@"fdd://"];

if (range.length!=0){
urlstr = [urlstr substringFromIndex:(range.location+range.length)];
NSString *method = urlstr;
SEL selctor = NSSelectorFromString(method);
[self performSelector:selctor withObject:nil];
return NO;
}

return YES;
}
#pragma clang diagnostic pop

- (void)alertShow{

}

- (void)actionsheetShow{

}

优点:实现JS调用OC代码,使用OC方法来处理JS的对应事件,可读性好

缺点:需要两端约定好方法,方法传多个参数不好处理

3.3 JavascriptCore框架:

CocoaChina iOS7新JavascriptCore框架介绍

hayageek Execute Javascript in iOS Applications

阿福的专栏 IOS7开发~JavascriptCore(一)

阿福的专栏 IOS7开发~JavascriptCore(二)

Kaitiren的专栏 JavascriptCore框架在iOS7中的对象交互和管理教程

http://trac.webkit.org/ JavascriptCore框架源码OC

http://opensource.apple.com/JavascriptCore框架源码C++

limlimlim的专栏 Javascript字典操作

j_akill的专栏 ios开发,Javascript直接调用oc代码而非通过改变url回调方式

jwzhangjie的专栏 OC与JS互相调用

JavascriptCore框架只要引入了5个文件,每个文件里都定义跟文件名对应的类:

  1. JSContext (提供着运行环境)

  2. JSValue (Javascript和Object-C之间互换的桥梁)

  3. JSManagedValue (将 JSValue 转为 JSManagedValue 类型后,可以添加到 JSVirtualMachine 对象中,这样能够保证你在使用过程中 JSValue 对象不会被释放掉,当你不再需要该 JSValue 对象后,从 JSVirtualMachine 中移除该 JSManagedValue 对象,JSValue 对象就会被释放并置空。

  4. JSVirtualMachine(JSVirtualMachine就是一个用于保存弱引用对象的数组,加入该数组的弱引用对象因为会被该数组 retain,所以保证了使用时不会被释放,当数组里的对象不再需要时,就从数组中移除,没有了引用的对象就会被系统释放。

  5. JSExport(神秘的语言穿梭机—JSExport协议)

HTML代码:



































JS代码:

function max(a, b){
return a>b?a:b;
}

function callBack(a, b){
var paramters = postParamters(20, 18); //执行OC的postParamters方法
var par1 = paramters['paramter1'];
var par2 = paramters['paramter2'];
paramters['paramter1'] = par1 + a;
paramters['paramter2'] = par2 + b;
return paramters;
}

function buttonClick2(title, message){
JavascriptAction(title, message);
}

var nativeManager; //需要全局申明
function buttonClick3(title, message){
nativeManager.doSomeThing('js to os' + title + ' ' , message);
}

function buttonClick4(title, message){
nativeManager.onePushSubmit('js to os' + title + ' ' , message);
}

function buttonClick5(s1, s2, s3, s4){
nativeManager.morePraramters(s1, s2, s3, s4);
}

iOS代码:

OC注入JS代码:

//OC注入JS代码,再获取JS代码中的变量
JSContext *iCOntext= [[JSContext alloc] init];
[iContext evaluateScript:@"var arr = [21, 7, 'harry up'];"]; //上下文执行该JS代码(变量)
JSValue *jsArr = iContext[@"arr"];
NSArray *ocArr = [jsArr toArray];
NSLog(@"JS Array: %@ \n OC Array: %@", jsArr, ocArr);

//OC注入JS代码,再获取JS代码中的方法
NSString *jsCode = @"function sum(a, b) {return a+b;}"; //定义JS方法
[iContext evaluateScript:jsCode]; //上下文执行该JS代码(方法)
JSValue *jsFunc = iContext[@"sum"]; //从上下文获取JS方法
JSValue *result = [jsFunc callWithArguments:@[@20, @18]]; //js方法传入参数并且返回结果
NSLog(@"result : %d", [result toInt32]);

OC通过JS文件注入JS代码

//OC注入JS代码(JS文件),再获取JS代码中的方法, (扩展)
NSString *path = [[NSBundle mainBundle] pathForResource:@"jsCore" ofType:@"js"];
NSString *pluginScript = [NSString stringWithContentsOfFile:path encoding:NSUTF8StringEncoding error:nil];
[iContext evaluateScript:pluginScript]; //上下文执行该JS代码(方法)
JSValue *jsFunc2 = iContext[@"max"]; //从上下文获取JS方法
JSValue *result2 = [jsFunc2 callWithArguments:@[@20, @18]]; //js方法传入参数并且返回结果
NSLog(@"result2 : %d", [result2 toInt32]);

//2、JS执行OC的postParamters方法
iContext[@"postParamters"] = ^(NSInteger a, NSInteger b){
NSDictionary *info = @{@"paramter1" : [NSNumber numberWithInteger:a + 20],
@"paramter2" : [NSNumber numberWithInteger:a + 18]};
return info;
};

//1、OC执行JS中callBack方法
JSValue *callBackFunc = iContext[@"callBack"];
JSValue *vv = [callBackFunc callWithArguments:@[@20, @18]];
NSDictionary *callBackDic = [vv toDictionary];
NSLog(@"callBackDic : %@", callBackDic);

OC和HTML上的JS交互

UIWebView *webView = [[UIWebView alloc] initWithFrame:self.view.bounds];
NSString *htmlPath = [[NSBundle mainBundle] pathForResource:@"redirect" ofType:@"html"];
NSURLRequest *rq = [NSURLRequest requestWithURL:[NSURL URLWithString:htmlPath]];
[webView loadRequest:rq];
webView.delegate = self;
[self.view addSubview:webView];

//此处通过当前webView的键获取到jscontext
JSContext *cOntext= [webView valueForKeyPath:@"documentView.webView.mainFrame.JavascriptContext"];
//通过context对JS上的每一个方法进行回调监听
context[@"JavascriptAction"] = ^(){
NSArray *args = [JSContext currentArguments];
for (JSValue *jsVal in args){
NSLog(@"%@", jsVal);
}
};

//通过对象对JS上的每一个方法进行回调监听
context[@"nativeManager"] = [NativeManager new];

NativeManager.h代码:

#import 
#import

@protocol NativeManagerExport

///<对于JS代码是:nativeManager.doSomeThing(s1, s2),JSExport协议根据JS方法,使用多个:进行分割
- (void)doSomeThing:(NSString *)someThing :(NSString *)s2;

///<对于JS代码是:nativeManager.onePushSubmit(s1, s2),JSExport协议根据JS方法中的大写字母进行方法分割
- (void)onePush:(NSString *)s1 submit:(NSString *)s2;

///<对于JS代码是:buttonClick5(s1, s2, s3, s4) 使用JSExport的提供的宏JSExportAs来处理
JSExportAs(morePraramters,
- (void)morePraramtersWithP1:(NSString *)p1 p2:(NSString *)p2 p3:(NSString *)p3 p4:(NSString *)p4
);
@end

@interface NativeManager : NSObject
@end

NativeManager.m代码:

#import "NativeManager.h"

@implementation NativeManager

- (void)doSomeThing:(NSString *)someThing :(NSString *)s2{
NSLog(@"使用OC对象来处理JS的点击事件");
NSLog(@"js 传递 : %@ %@", someThing, s2);
}

- (void)onePush:(NSString *)s1 submit:(NSString *)s2{
NSLog(@"使用OC对象来处理JS的点击事件2");
NSLog(@"js 传递 : %@ %@", s1, s2);
}

- (void)morePraramtersWithP1:(NSString *)p1 p2:(NSString *)p2 p3:(NSString *)p3 p4:(NSString *)p4{
NSLog(@"使用OC对象来处理JS的点击事件3");
NSLog(@"js 传递 : %@ %@ %@ %@", p1, p2, p3, p4);
}

@end

iOSlog日志:

JSToOCDemo[28592:1486323] JS Array: 21,7,harry up 
OC Array: (
21,
7,
"harry up"
)

JSToOCDemo[54490:1751112] result : 38
JSToOCDemo[60727:1932210] callBackDic : {
paramter1 = 60;
paramter2 = 56;
}

JSToOCDemo[28592:1486529] JS点击方法传递参数标题
JSToOCDemo[28592:1486529] JS点击方法传递参数信息

优点:实现JS调用OC代码,使用OC类来管理JS的对应方法,模块化清晰,能处理多个带参数方法

缺点:内存问题需要自己处理好,支持iOS7以上的项目才能使用

3.4 EasyJSWebView第三方代码:

github dukeland/EasyJSWebView

开源中国-珲少 IOS NSInvocation应用与理解

iteye-啸笑天 Objective C运行时(runtime)技术的几个要点总结

HTML代码:


iOS代码:

EasyJSWebView *eWebView = [[EasyJSWebView alloc] initWithFrame:self.view.bounds];
[self.view addSubview:eWebView];

EJSWNativeManager *manager = [EJSWNativeManager new];
//将OC管理类对象manager和JS的全局对象easyJSManager绑定
[eWebView addJavascriptInterfaces:manager WithName:@"easyJSManager"];
eWebView.delegate = self;

NSString *htmlPath = [[NSBundle mainBundle] pathForResource:@"redirect" ofType:@"html"];
NSURLRequest *rq = [NSURLRequest requestWithURL:[NSURL URLWithString:htmlPath]];
[eWebView loadRequest:rq];

EJSWNativeManager.h代码:

#import 

@interface EJSWNativeManager : NSObject
- (void)saveUserInfo:(NSString *)name :(NSString *)password;
@end

EJSWNativeManager.m代码:

#import "EJSWNativeManager.h"

@implementation EJSWNativeManager
- (void)saveUserInfo:(NSString *)name :(NSString *)password{
NSLog(@"EasyJSWebView点击事件带参数");
NSLog(@"js 传递 : %@ %@", name, password);
}
@end

EasyJSWebView源码分析:在UIWebViewDelegate代理的webViewDidStartLoad方法中,根据 [eWebView addJavascriptInterfaces:manager WithName:@”easyJSManager”] 方法的OC管理类对象manager和JS的全局对象easyJSManager在document创建新的子节点,该子节点的href对应的链接是由easyJSManager和manager的方法按照一定方式拼接,在点击HTML上的 “EasyJSWebView点击事件带参数”按钮时,button的onClick事件:Javascript:easyJSManager.saveUserInfo会重新 调用OC的webView: shouldStartLoadWithRequest: navigationType:方法,这个时候再根据一定方式拼接来找到该点击事件所附带的信息。

优点:实现JS调用OC代码,使用OC类来管理JS的对应方法,模块化清晰

缺点:必须继承EasyJSWebView,这使得它和第三方优秀的webview不好同时使用

项目下载地址


推荐阅读
  • Voicewo在线语音识别转换jQuery插件的特点和示例
    本文介绍了一款名为Voicewo的在线语音识别转换jQuery插件,该插件具有快速、架构、风格、扩展和兼容等特点,适合在互联网应用中使用。同时还提供了一个快速示例供开发人员参考。 ... [详细]
  • 本文介绍了如何使用PHP向系统日历中添加事件的方法,通过使用PHP技术可以实现自动添加事件的功能,从而实现全局通知系统和迅速记录工具的自动化。同时还提到了系统exchange自带的日历具有同步感的特点,以及使用web技术实现自动添加事件的优势。 ... [详细]
  • Monkey《大话移动——Android与iOS应用测试指南》的预购信息发布啦!
    Monkey《大话移动——Android与iOS应用测试指南》的预购信息已经发布,可以在京东和当当网进行预购。感谢几位大牛给出的书评,并呼吁大家的支持。明天京东的链接也将发布。 ... [详细]
  • 使用nodejs爬取b站番剧数据,计算最佳追番推荐
    本文介绍了如何使用nodejs爬取b站番剧数据,并通过计算得出最佳追番推荐。通过调用相关接口获取番剧数据和评分数据,以及使用相应的算法进行计算。该方法可以帮助用户找到适合自己的番剧进行观看。 ... [详细]
  • 如何实现织梦DedeCms全站伪静态
    本文介绍了如何通过修改织梦DedeCms源代码来实现全站伪静态,以提高管理和SEO效果。全站伪静态可以避免重复URL的问题,同时通过使用mod_rewrite伪静态模块和.htaccess正则表达式,可以更好地适应搜索引擎的需求。文章还提到了一些相关的技术和工具,如Ubuntu、qt编程、tomcat端口、爬虫、php request根目录等。 ... [详细]
  • 本文详细介绍了GetModuleFileName函数的用法,该函数可以用于获取当前模块所在的路径,方便进行文件操作和读取配置信息。文章通过示例代码和详细的解释,帮助读者理解和使用该函数。同时,还提供了相关的API函数声明和说明。 ... [详细]
  • 本文介绍了数据库的存储结构及其重要性,强调了关系数据库范例中将逻辑存储与物理存储分开的必要性。通过逻辑结构和物理结构的分离,可以实现对物理存储的重新组织和数据库的迁移,而应用程序不会察觉到任何更改。文章还展示了Oracle数据库的逻辑结构和物理结构,并介绍了表空间的概念和作用。 ... [详细]
  • 本文分享了一个关于在C#中使用异步代码的问题,作者在控制台中运行时代码正常工作,但在Windows窗体中却无法正常工作。作者尝试搜索局域网上的主机,但在窗体中计数器没有减少。文章提供了相关的代码和解决思路。 ... [详细]
  • 本文介绍了使用Java实现大数乘法的分治算法,包括输入数据的处理、普通大数乘法的结果和Karatsuba大数乘法的结果。通过改变long类型可以适应不同范围的大数乘法计算。 ... [详细]
  • 本文是一位90后程序员分享的职业发展经验,从年薪3w到30w的薪资增长过程。文章回顾了自己的青春时光,包括与朋友一起玩DOTA的回忆,并附上了一段纪念DOTA青春的视频链接。作者还提到了一些与程序员相关的名词和团队,如Pis、蛛丝马迹、B神、LGD、EHOME等。通过分享自己的经验,作者希望能够给其他程序员提供一些职业发展的思路和启示。 ... [详细]
  • CSS3选择器的使用方法详解,提高Web开发效率和精准度
    本文详细介绍了CSS3新增的选择器方法,包括属性选择器的使用。通过CSS3选择器,可以提高Web开发的效率和精准度,使得查找元素更加方便和快捷。同时,本文还对属性选择器的各种用法进行了详细解释,并给出了相应的代码示例。通过学习本文,读者可以更好地掌握CSS3选择器的使用方法,提升自己的Web开发能力。 ... [详细]
  • 知识图谱——机器大脑中的知识库
    本文介绍了知识图谱在机器大脑中的应用,以及搜索引擎在知识图谱方面的发展。以谷歌知识图谱为例,说明了知识图谱的智能化特点。通过搜索引擎用户可以获取更加智能化的答案,如搜索关键词"Marie Curie",会得到居里夫人的详细信息以及与之相关的历史人物。知识图谱的出现引起了搜索引擎行业的变革,不仅美国的微软必应,中国的百度、搜狗等搜索引擎公司也纷纷推出了自己的知识图谱。 ... [详细]
  • 如何用UE4制作2D游戏文档——计算篇
    篇首语:本文由编程笔记#小编为大家整理,主要介绍了如何用UE4制作2D游戏文档——计算篇相关的知识,希望对你有一定的参考价值。 ... [详细]
  • http:my.oschina.netleejun2005blog136820刚看到群里又有同学在说HTTP协议下的Get请求参数长度是有大小限制的,最大不能超过XX ... [详细]
  • 【MEGA DEAL】Ruby on Rails编码训练营(97%折扣)限时特惠!
    本文介绍了JCG Deals商店提供的Ruby on Rails编码训练营的超值优惠活动,现在只需29美元即可获得,原价为$1,296。Ruby on Rails是一种用于Web开发的编程语言,即使没有编程或网页设计经验,也能在几分钟内构建专业的网站。该训练营共有6门课程,包括使用Ruby on Rails进行BDD的课程,使用RSpec 3和Capybara等。限时特惠,机会难得,赶快行动吧! ... [详细]
author-avatar
小鱼儿太疯癫
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有