android js交互 数组,Android WebView —— Java 与 JavaScript 交互总结
相比于 Native App 和 Web App,Hybrid App 凭借其迭代灵活、控制自如、多端同步的优势在应用市场上越发显得优胜,主要得力于,其将变更频繁的部分产品功能使用 H5 开发并在客户端中借助 WebView 控件嵌入应用当中。所以,开发中我们总会遇到原生 Java 代码与网页中的 Js 代码之间相互调用从而产生的交互问题。
Java 与 Js 彼此调用的前提是设置 WebView 支持 JavaScript 功能:
mWebView.getSettings().setJavaScriptEnabled(true);复制代码
Java 调用 Js
第一步,在网页中使用 Js 定义提供给 Java 访问的方法,就像普通方法定义一样,如:
function javaCallJs(message){
alert(message);
}
复制代码
第二步,在 Java 代码中按照 "javascript:XXX" 的 Url 格式使用 WebView 加载访问即可:
mWebView.loadUrl("javascript:javaCallJs(" + "'Message From Java'" + ")");复制代码注意:String 类型的参数需要使用单引号 “'” 包裹,数组类型的参数则不用,如:javascript:javaCallJs([01, 02, 03]),其他复杂类型的参数可以转换为 Json 字符串的形式传递。
Js 调用 Java
第一步,在 Java 对象中定义 Js 访问的方法,如:
@JavascriptInterface
public void jsCallJava(String message){
Toast.makeText(this, message, Toast.LENGTH_SHORT).show();
}复制代码注意事项:提供给 Js 访问的属性和方法必须定义为 public 类型,并且添加注解 @JavascriptInterface。在 API 17 及更高版本的系统中,任何暴露给 Js 访问的 Java 接口都需要添加这个注解,否则会报异常:Uncaught TypeError: Object [object Object] has no method 'XXX'。系统这种做法也是为了降低应用的安全隐患,因为在之前的版本中,Js 可以通过反射的方式访问注入 WebView 中的 Java 对象的 public 类型 field 和 method,从而随意修改宿主程序。
第二步,将提供给 Js 访问的接口内容所属的 Java 对象注入 WebView 中:
mWebView.addJavascriptInterface(MainActivity.this, "main");复制代码addJavascriptInterface(Object object, String name) 参数说明:object 表示 Js 访问的接口内容所在的 Java 对象;name 表示 Js 调用 Java 代码时的接口名称,与 Js 中的调用保持一致即可。
第三步,Js 按照指定的接口名访问 Java 代码,有如下两种写法:
Js Call Java
复制代码
这里简单提供一个可供测试的 Html 网页和 Activity 代码:
test.html:
alert(message);
}
Js Call Java
复制代码
MainActivity.java:
public class MainActivity extends AppCompatActivity{
private WebView mWebView;
@Override
protected void onCreate(Bundle savedInstanceState){
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
Toolbar mToolbarTb = (Toolbar) findViewById(R.id.tb_toolbar);
setSupportActionBar(mToolbarTb);
mWebView = (WebView) findViewById(R.id.webview);
mWebView.getSettings().setJavaScriptEnabled(true);
mWebView.loadUrl("file:///android_asset/test.html");
mWebView.addJavascriptInterface(MainActivity.this, "main");
mWebView.setWebChromeClient(new WebChromeClient() {
@Override
public boolean onJsAlert(WebView view, String url, String message, JsResult result){
return super.onJsAlert(view, url, message, result);
}
});
}
public void javaCallJs(View v){
mWebView.loadUrl("javascript:javaCallJs(" + "'Message From Java'" + ")");
}
@JavascriptInterface
public void jsCallJava(String message){
Toast.makeText(this, message, Toast.LENGTH_SHORT).show();
}
@Override
public boolean onCreateOptionsMenu(Menu menu){
getMenuInflater().inflate(R.menu.search, menu);
return super.onCreateOptionsMenu(menu);
}
}复制代码
效果图:
注意:无论是 Java 调用 Js 还是 Js 调用 Java,只能通过参数传递数据,而无法获取彼此方法的返回值!解决方案就是额外添加一层回调来达到这个目的。比如 Java 调用 Js 的方法,Js 计算结束所得结果不能通过 return 语句返回给 Java 调用者,而是再回调 Java 的另一个方法,通过传参的形式传递给 Java。
注意事项
1.使用 loadUrl() 方法实现 Java 调用 Js 功能时,必须放置在主线程中,否则会发生崩溃异常。比如修改上面的代码:
new Thread(new Runnable() {
@Override
public void run(){
mWebView.loadUrl("javascript:javaCallJs(" + "'Message From Java'" + ")");
}
}).start();复制代码
运行时会得到如下 logcat 异常信息:
java.lang.RuntimeException: java.lang.Throwable: A WebView method was called on thread 'Thread-18022'. All WebView methods must be called on the same thread.复制代码
如果真的在子线程中遇到调用 Js 的功能,也要将其转换到主线程中去:
mWebView.post(new Runnable() {
@Override
public void run(){
mWebView.loadUrl("javascript:javaCallJs(" + "'Message From Java'" + ")");
}
});复制代码
2.Js 调用 Java 方法时,不是在主线程 (Thread Name:main) 中运行的,而是在一个名为 JavaBridge 的线程中执行的,通过如下代码可以测试:
@JavascriptInterface
public void jsCallJava(String message){
Log.i("thread", Thread.currentThread().getName());
Toast.makeText(this, message, Toast.LENGTH_SHORT).show();
}复制代码
所以这里需要注意的是,当 Js 调用 Java 时,如果需要 Java 继续回调 Js,千万别在 JavascriptInterface 方法体中直接执行 loadUrl() 方法,而是像前面一样进行线程切换操作。
3.代码混淆时,记得保持 JavascriptInterface 内容,在 proguard 文件中添加如下类似规则 (有关类名按需修改):
keepattributes *Annotation*
keepattributes JavascriptInterface
-keep public class com.mypackage.MyClass$MyJavaScriptInterface
-keep public class * implements com.mypackage.MyClass$MyJavaScriptInterface
-keepclassmembers class com.mypackage.MyClass$MyJavaScriptInterface{
;
}复制代码
Url 拦截
除了上面这种 Java 与 Js 互调方法的方式,还可以利用 WebView 拦截 Url 的方式实现原生应用与 H5 之间的交互动作。通过 WebViewClient 提供的接口拦截网页内诸如二级跳转的 Url 链接,便可以进行业务逻辑上的判断处理、Url 参数传递等功能,如:
mWebView.setWebViewClient(new WebViewClient(){
@Override
public boolean shouldOverrideUrlLoading(WebView view, WebResourceRequest request){
// request.getUrl()
return super.shouldOverrideUrlLoading(view, request);
}
});复制代码注意:过去常用的 shouldOverrideUrlLoading(WebView view, String url) 方法已经被废弃。
参考使用
通过 Java 与 Js 之间的交互可以做很多事情,比如获取网页中的图片,利用原生控件予以展示,类似响应微信公众号文章中的图片点击事件。参考代码如下:
public class MainActivity extends AppCompatActivity{
private WebView mWebView;
@Override
protected void onCreate(Bundle savedInstanceState){
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
mWebView = (WebView) findViewById(R.id.webview);
mWebView.getSettings().setJavaScriptEnabled(true);
mWebView.loadUrl("https://www.taobao.com/");
mWebView.addJavascriptInterface(new MyJavascriptInterface(), "imageClick");
mWebView.setWebViewClient(new MyWebViewClient());
}
/**
* 遍历 标签, 添加图片点击事件, 将图片 Url 地址回调给 Java 方法
*/
private void addImageClickListner(){
mWebView.loadUrl("javascript:(function(){" +
"var objs = document.getElementsByTagName(\"img\"); " +
"for(var i=0;i
"{"
+ " objs[i].οnclick=function() " +
" { "
+ " window.imageClick.openImage(this.src); " +
" } " +
"}" +
"})()");
}
public class MyJavascriptInterface{
public MyJavascriptInterface(){
}
@android.webkit.JavascriptInterface
public void openImage(String imageUrl){
Log.i("imageUrl", imageUrl);
// TODO 获取图片地址后, 通过原生控件 ImageView 展示, 添加缩放、保存等功能
}
}
private class MyWebViewClient extends WebViewClient{
@Override
public boolean shouldOverrideUrlLoading(WebView view, WebResourceRequest request){
return super.shouldOverrideUrlLoading(view, request);
}
@Override
public void onPageFinished(WebView view, String url){
super.onPageFinished(view, url);
addImageClickListner();
}
}
}复制代码
最后
本文由 亦枫 创作并首发于 亦枫的个人博客 ,同步授权微信公众号:技术鸟(NiaoTech)。
欢迎各种形式地交流与转载,注明作者及出处即可。
本文标题为: Android WebView —— Java 与 JavaScript 交互总结
android js交互 数组,Android WebView —— Java 与 JavaScript 交互总结相关推荐
- Android入门——WebView使用及Java和JavaScript交互小结
文章大纲 引言 一.WebView 概述 二.WebView的常用的方法 1.构造方法 2.其他常用方法 3.WebSettings常用的方法 三.WebView的应用 1.使用WebView的基本步 ...
- android java 调用js,Android中Java和JavaScript交互实例
Android提供了一个很强大的WebView控件用来处理Web网页,而在网页中,JavaScript又是一个很举足轻重的脚本.本文将介绍如何实现Java代码和Javascript代码的相互调用. 如 ...
- android 变量定义数组,android 创建数组
一: private String[] data = new String[]{"Hello", "jike", "world"}; 二: ...
- android文件存储数组,Android面试简录——文件存储
* SharedPreferences 请描述Android SDK支持哪些文件存储技术? 1.SharedPreferences保存key-value类型的数据 2.流文件存储(openFileOu ...
- android js gps定位,Android中的定位Demo
可以按照这上面的流程去申请key值.在获取SHA1值的时候,推荐使用给的第一种方法(在cmd命令中获取,我之前使用的是从Eclipse> Window > Preferences > ...
- qt与js html进行数据传递,QT与javascript交互数据的实现
一.数据从QT流向JS 1.QT调用JS的函数,JS通过形参获得QT的值 2.JS调用QT的函数,QT函数的返回值进入JS 二.数据从JS流向QT 1.JS调用QT的函数,QT通过形参获得JS的值 2 ...
- android 定义固定数组,Android 图片数组定义和读取
位置:packages/apps/Launcher2 1.图片数组定义.资源读取 如果有多张图片,这些图片的使用与顺序无关,可以采取这种方式. drawable-nodpi中有3张图片,wallpap ...
- android 输出字节数组,Android蓝牙通信字节数组的数据类型转换 求教!
该楼层疑似违规已被系统折叠 隐藏此楼查看此楼 单片机和安卓手机进行蓝牙通信,发送电压数据到手机,恰好每次8位数据(1字节)传给上位机.采用以下代码获得16进制字符输出,以下是蓝牙通信接收并显示数据部分 ...
- android 数据库 字节数组,Android上的GreenDAO:字节数组作为主键/构建包含字节数组属性Where子句的查询...
关于字节[]又名在greendao BLOB: 看着de.greenrobot.dao.query.WhereCondition.PropertyCondition.checkValueForType ...
最新文章
- 5-django——验证码,中间件
- php访问参数错误,phpcms参数错误怎么办
- 50万买只波士顿动力机器狗,只为训练它“尿”啤酒,还让女朋友尝尝
- 同质化软件“吃掉”创新 企业因何深陷罗网?
- RocketMQ在面试中那些常见问题及答案+汇总
- Java操作HBase 2.0.5:创建表代码示例
- 从KDD 2018最佳论文看Airbnb实时搜索排序中的Embedding技巧
- 【PAT甲级】1048 Find Coins (25 分) C++ 全部AC
- dom4j Java解析xml应用
- HTML map元素
- java file.length 单位_Java.io.File.length()返回0
- Python的subprocess模块(一)
- 逻辑回归算法及其实现
- mybatis在配置mapper.xml时出现 tag name excepted 问题
- python爬虫由浅入深8---基于正则表达式查询的淘宝比价定向爬虫
- 网络编程中同步与异步,IO阻塞与非阻塞总结
- Nvidia Tesla A100 驱动安装方法
- HTML5实现大文件上传
- [Java开发]搭建人力资源管理系统——简历管理模块(附带下载链接)
- 高考状元、奥赛金牌,清华姚班00后新生来了
热门文章
- 基于Gsoap 的ONVIF C++ 库
- Windows服务创建及安装
- 【Android】spannableStringBuilder
- WindowsXP 下搭建PHP环境(笔记)
- python3librequest_python3.x學習之urilib.request簡單學習
- 连接Oracle时报错ORA-12541: TNS: 无监听程序
- case when..then
- C#基于websocket-sharp实现简易httpserver(封装)
- Apache性能诊断与调优
- 【Java从0到架构师】Spring - AOP