百度360必应搜狗淘宝本站头条
当前位置:网站首页 > 技术文章 > 正文

Objective-C与JavaScript交互的那些事

myzbx 2024-12-12 13:35 18 浏览

最近公司的运营瞎搞了个活动,其活动要服务端提供数据支持,web前端在微信公众账号内作为主要的运营阵地,而iOS、Android要提供相应的入口及页面进行配合。一个活动,动用了各个端的程序猿。而在这里面技术方面主要就是涉及到web端和服务端的交互,web前端和iOS、Android的交互。本人作为一个iOS开发者,今天就聊聊web、iOS、Android三端的交互,其实在说明白一点就是方法的互相调用而已。这里主要讲解iOS。Android会稍微提一下,仅作参考。

此篇文章的逻辑图

图0-0 此篇文章的逻辑图

概述

iOS原生应用和web页面的交互大致上有这几种方法iOS7之后的JavaScriptCore、拦截协议、第三方框架WebViewJavaScriptBridge、iOS8之后的WKWebView在这里主要讲解JavaScriptCore和拦截协议这两种办法。WebViewJavaScriptBridge是基于拦截协议进行的封装。学习成本相对JavaScriptCore较高,使用也不如JavaScriptCore方便本文不做叙述。WKWebView是iOS8之后推出的,还没有成为主流使用,所以本篇文章也不做详细叙述。

Objective-C执行JavaScript代码

相关方法

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

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

相关应用

用这些方法去执行大段的JavaScript代码是没什么必要的,但是有些小场景用起来还是比较顺手和实用的,列举两个例子作为参考:

//?获取当前页面的title
NSString?*title?=?[webview?stringByEvaluatingJavaScriptFromString:@"document.title"];

//?获取当前页面的url
NSString?*url?=?[webview?stringByEvaluatingJavaScriptFromString:@"document.location.href"];

JavaScriptCore

iOS7之后苹果推出了JavaScriptCore这个框架,从而让web页面和本地原生应用交互起来非常方便,而且使用此框架可以做到Android那边和iOS相对统一,web前端写一套代码就可以适配客户端的两个平台,从而减少了web前端的工作量。

web前端

在三端交互中,web前端要强势一些,一切传值、方法命名都按web前端开发人员来定义,让另外两端去做适配。在这里以调用摄像头和分享为例来详细讲解,测试网页代码取名为test.html,其代码内容如下:

test.html代码内容(因识别问题,用方括号替换了代码中的尖括号)

[!DOCTYPE?html]
[html]
[head]
????[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":?"标题",?"desc":?"内容",?"shareUrl":?"http://www.jianshu.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代码解释

可能有些同学对web前端的一些知识不太熟悉,稍微对这段代码做下解释,先说Toyun是iOS和Android这两边在本地要注入的一个对象【参考下面iOS的代码更容易明白】,充当原生应用和web页面之间的一个桥梁。页面上定义了两个按钮名字分别为CallCamera和Share。点击CallCamera会通过Toyun这个桥梁调用本地应用的方法- (void)callCamera,没有传参;而点击Share会先调用本文件中的JavaScript方法callShare这里将要分享的内容格式转成JSON字符串格式(这样做是为了适配Android,iOS可以直接接受JSON对象)然后再通过Toyun这个桥梁去调用原生应用的- (void)share:(NSString *)shareInfo方法这个是有传参的,参数为shareInfo。而下面的两个方法为原生方法调用后的回调方法,其中picCallback为获取图片成功的回调方法,并且传回拿到的图片photos;shareCallback为分享成功的回调方法。

iOS

iOS这边根据前端定义的方法名来写代码,但是有些时候web前端会让我们定义,但是我们定义好之后他又要修改,这时候就会很烦啊。所以碰到三端交互的时候最好就是让web前端去定义方法名,iOS和Android根据web前端定义好的去写代码。JavaScriptCore中web页面调用原生应用的方法可以用Delegate或Block两种方法,此文以按Delegate讲解。

JavaScriptCore中类及协议:

  • JSContext:给JavaScript提供运行的上下文环境

  • JSValue:JavaScript和Objective-C数据和方法的桥梁

  • JSManagedValue:管理数据和方法的类

  • JSVirtualMachine:处理线程相关,使用较少

  • JSExport:这是一个协议,如果采用协议的方法交互,自己定义的协议必须遵守此协议

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");
????//?获取到照片之后在回调js的方法picCallback把图片传出去
????JSValue?*picCallback?=?self.jsContext[@"picCallback"];
????[picCallback?callWithArguments:@[@"photos"]];
}

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

@end

ViewController中的代码解释

自定义JSObjcDelegate协议,而且此协议必须遵守JSExport这个协议,自定义协议中的方法就是暴露给web页面的方法。在webView加载完毕的时候获取JavaScript运行的上下文环境,然后再注入桥梁对象名为Toyun,承载的对象为self即为此控制器,控制器遵守此自定义协议实现协议中对应的方法。在JavaStript调用完本地应用的方法做完相对应的事情之后,又回调了JavaStript中对应的方法,从而实现了web页面和本地应用之间的通讯。

JavaScriptCore使用注意

JavaStript调用本地方法是在子线程中执行的,这里要根据实际情况考虑线程之间的切换,而在回调JavaScript方法的时候最好是在刚开始调用此方法的线程中去执行那段JavaStript方法的代码,我在实际运用中开始没注意,就被坑惨了啊。什么,说的太绕,看下面的代码解释:

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

运行效果

运行效果如图3-1所示

图3-1 运行效果

拦截协议

拦截协议这个适合一些比较简单的一些情况,不需要引入什么框架,只需要web前端配合一下就好。但是在具体调用哪一个方法上,以及在传值的时候可能会有些不方便,而且调用完后无法在回调JavaScript的方法。

web前端

test.html中的代码(因识别问题,用方括号替换了代码中的尖括号)

[!DOCTYPE?html]
[html]
[head]
????[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点击之后调用自己的callCamera方法,window.location.href这里是改变主窗口的指向从而马上发出一个链接为Toyun://callCamera请求,而想要传给原生应用的参数也可已包含到此请求中,而在iOS方法中我们要拦截这个请求,根据请求内容去判断JavaStript想要做的事情,从而实现web页面和本地应用之间的交互。

iOS

iOS对应的代码

-?(BOOL)webView:(UIWebView?*)webView?shouldStartLoadWithRequest:(NSURLRequest?*)request?navigationType:(UIWebViewNavigationType)navigationType
{
????NSString?*url?=?request.URL.absoluteString;
????if?([url?rangeOfString:@"toyun://"].location?!=?NSNotFound)?{?
????????//?url的协议头是Toyun
????????NSLog(@"callCamera");
????????return?NO;
????}
????return?YES;
}

iOS对应的代码的解释

在webView的代理方法中去拦截自定义的协议Toyun://如果是此协议则据此判断JavaStript想要做的事情,调用原生应用的方法,这些都是提前约定好的,同时阻止此链接的跳转。

总结

随着手机硬件的配置越来越强大和HTML5的兴起,一个App完全可以由web页面来写。现在已经有部分应用这么干了,我是遇见过的,如古诗文网。尽管比较少但是web页面和本地应用的交互不论是iOS还是Android都是会有遇到的。iOS我还是比较推荐JavaScriptCore,这样三端可以相对统一起来,写的时候都比较简单。随着时间的推移iOS8推出的WKWebView会逐渐成为主流,这个的功能更强大。拦截协议也只能说用到比较简单的一些情况吧,复杂的情况处理相互之间参数的传递还是比较麻烦的,而且这个不能回调JavaScript的方法,确实喜欢拦截协议的同学可以研究WebViewJavaScriptBridge这个第三方库。对于Android本人也就是略知皮毛而已,就不班门弄斧了,对于一些Android开发者来说,可以看地第一段的test.html这个页面的写法完全是可以适配Android的。

参考

相关推荐

Fabric.js使用说明详解(fabric nodejs)

Fabric介绍简介:Fabric是一款基于HTML5Canvas的开源绘图库,它提供了丰富的API和工具,可以轻松地创建交互式的绘图应用程序和游戏。功能特点:介绍Fabric的核心功能,如图形绘制...

如何才能快速将照片变成漫画?(如何才能快速将照片变成漫画风格)

本文分享8个超级详细的照片变漫画教程,让你秒变绘画大师,跟着教程就能亲手将自己的照片制作成精美的漫画,快来一起试试吧~1、PS一款专业的图像处理软件,具备出色的图像处理功能,提供了强大的编辑工具和广...

很少人知道的20个最好用的免费设计软件

不是每个人都能花重金去购置昂贵的软件来装备自己,特别是在刚开始涉足设计这个行业的新手们。我们知道adobe系列软件正版是收费的,而且很贵,虽然说我们国内已经能破解使用,很多人都认为adobe的那些软件...

13个免费的信息图表制作软件推荐(如何制作信息表)

制作信息图表涉及大量的工作,如:数据收集,数据排列,选择和规划信息图表的格式和设计信息图表。在设计信息图表时,选择适当的制图软件是一个非常艰巨的任务。下面为大家推荐13个最好的免费信息图表制作软件。A...

怎么转换图片的格式?(怎么转换图片的格式)

我们日常用到最多的图片格式是JPG/JPEG格式的,其他常见的图片格式包括有bmp,png,tif,gif,svg,cdr,ai,raw,webp等,由于图片的使用场景不同,所以要求的具体格式也会有所...

Adobe的各种图形处理软件分别是做什么的

Adobe公司的Ps,Pr,Ae,An,Ai软件分别的用来做什么的?PS:AdobePhotoshop的简称,用于图像处理、编辑、通道、图层、路径综合运用,图像色彩的校正,各种特效滤镜的使用、特效字...

前端开发中,对图片的优化技巧有哪些?

按照先后顺序有以下:1.去掉无意义的修饰。嗯,我会瞎说吗?除了内容图片,其他的图片的作用是修饰,也就是对于传达信息来说并非本质性的。最大的优化就是压根不要图片!所以在优化之前要做的,首先是确认设计,...

Adobe软件的最佳Linux替代品有哪些?这几款工具不容错过

  如果你是一名平面设计工作者,那么估计很多与你职业相同或者是喜欢并正在学习平面设计的小伙伴们,而如果恰巧你们都是Linux用户的话,那么估计很多的小伙伴都在寻找Adobe的Linux替代品吧。但是,...

精心收集几个免费绘图、修图与照片编辑软件分享一下

本次为大家介绍几款绘图、修图与照片编辑软件,都可以免费授权个人与教育使用,有些开源软件甚至可以在工作上商业使用。其中大多数的绘图软件都是跨平台的,可以在Windows或Mac上安装,对个人照片...

UI设计入门干货!八大软件+技能+素材网站

随着互联网行业的发展,UI设计师越来越多的被提及,UI设计师大火,需求岗位越来越多,也有越来越多的人转行投身UI设计师。UI设计是什么?一般所说的UI设计多指UI视觉设计,主要负责APP、Web、H5...

干货!一文读懂10种主流的图片格式

JPG、PNG、GIF,这些在我们生活中常见的图片格式,你真的了解它们吗?你知道除了这3种图片格式外,还有十多种主流的图片格式吗?每一种图片格式都有自己的特点和适用场景,选择正确的图片类型不仅能提升视...

最佳设计:A-Frame,阅读进度指示器,Colorify.js,交互式讲故事

最佳设计和开发好东西的综述以及一些新版本-第44期(2015年最后一期),包括A-Frame,阅读进度指示器,Colorify.js,交互式讲故事的动画地图路径,简单的Ionic侧菜单过渡等。摇滚...

当爆火的“粘土风”吹进铁路......

展播开始啦⑨<animateattributeName="opacity"begin="0s"dur="0.01"fill="freeze"from="1"to="1"/>&l...

日日是好日:书法艺术文字T恤定制,陶冶性情、现代生活新平衡

书法,心情和思想都融入文字的意境当中,对眼前或身边发生的不愉快事情视而不见、听而不闻,从而进入既轻松又舒适的状态,没有了妄念和烦恼,精神获得享受。本文节选自《DIYSKU个性化定制设计按需印刷行业出海...

从默默无闻到无可替代,Photoshop背后藏着多少不为人知的故事 ?

1990年2月推出了photoshop1.0。当时Photoshop只能在Mac计算机上运行,功能上也只有“工具”面板和少量的滤镜。1991年2月推出了photoshop2.0。该版本发行引发了桌...