Android WebView 与JS的数据交互

简介: 转自 :http://blog.csdn.net/cappuccinolau/article/details/8262821 关于WebView 我们知道目前android市场上的一些应用采用的开发方式大致分为三种:Native App、Web App、Hybrid App。本文主要是Hybrid App中实现的主要技术native组件与js的数据交互的理解以及实现

转自 :http://blog.csdn.net/cappuccinolau/article/details/8262821


关于WebView

我们知道目前android市场上的一些应用采用的开发方式大致分为三种:Native AppWeb AppHybrid App。本文主要是Hybrid App中实现的主要技术native组件与js的数据交互的理解以及实现。

 

Android API中提供了WebView组件来实现对html的渲染。所谓的HybridApp开发方式即是汇集了HTML5CSS3jS的相关开发技术,以及数据交换格式json/XML。这显然是Web开发工程师的技能。正是因为如此,众多中小企业选择了这种开发方式来减少对android开发工程师的过度依赖,至于这三种开发方式的比较与优劣不在本文考虑之列。

 

有了WebView这个组件,Android应用开发技术也就转嫁到htmljava数据交互上来。说白了就是jsWebView的数据交互,这就是本文所要讨论的。

WebViewjs的数据交互

1.        WebView中载入静态页面

 

WebView添加到应用中。和原生控件一样,在layout引入WebView控件。代码片段如下:

[html]  view plain copy
  1. <?xml version="1.0" encoding="utf-8"?>  
  2. <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"  
  3.     android:id="@+id/linearLayout"  
  4.     android:layout_width="match_parent"  
  5.     android:layout_height="match_parent"  
  6.     android:background="#000"  
  7.     android:orientation="horizontal" >  
  8. <WebView  
  9.     android:id="@+id/webview"  
  10.     android:layout_width="match_parent"  
  11.     android:layout_height="match_parent"   
  12.     />  
  13. </LinearLayout>  

载入页面:

 

[java]  view plain copy
  1. webView = (WebView) findViewById(R.id.webview);  
  2. webView.loadUrl("file:///file:///android_asset/page.html");  

page.html存储在工程文件的assets根目录下。

2.        引入jquery mobile

引入js框架让我们编写的html页面更接近于原生控件的显示效果。目前主流的移动应用js框架有:jquery mobilesencha touchjquery mobilesencha touch的选型不在本文讨论范围)。本文选择使用jquery mobile

 

首先,在webview添加对js的支持:

[java]  view plain copy
  1. WebSettings setting = webView.getSettings();  
  2. setting.setJavaScriptEnabled(true);//支持js  

增加对中文的支持:

[java]  view plain copy
  1. WebSettings setting = webView.getSettings();  
  2. setting.setDefaultTextEncodingName("GBK");//设置字符编码  

设置页面滚动条风格:

[java]  view plain copy
  1. webView.setScrollBarStyle(0);//滚动条风格,为0指滚动条不占用空间,直接覆盖在网页上  

jquery mobile提供的标准页面模板TemplateForJQuery.html

[html]  view plain copy
  1. <!DOCTYPE html>   
  2. <html>   
  3.     <head>   
  4.     <title>Page Title</title>   
  5.       
  6.     <meta name="viewport" content="width=device-width, initial-scale=1">   
  7.   
  8.     <link rel="stylesheet" href="css/jquery.mobile-1.1.1.min.css" />  
  9.     <script src="js/jquery.js"></script>  
  10.     <script src="js/jquery.mobile-1.1.1.min.js"></script>  
  11. </head>   
  12. <body>   
  13.   
  14. <div data-role="page">  
  15.   
  16.     <div data-role="header">  
  17.         <h1>Page Title</h1>  
  18.     </div><!-- /header -->  
  19.   
  20.     <div data-role="content">   
  21.         <p>Page content goes here.</p>        
  22.     </div><!-- /content -->  
  23.   
  24.     <div data-role="footer">  
  25.         <h4>Page Footer</h4>  
  26.     </div><!-- /footer -->  
  27. </div><!-- /page -->  
  28.   
  29. </body>  
  30. </html>  

页面依赖的js库、css等均放在assets目录下,目录组织结构如下:

运行应用后的截图:

下面是button 的截图,与原生控件没什么明显区别,有种以假乱真的感觉:

3.        良好的用户体验

运行我们的应用发现,在拥有大量js的页面被载入时,一直处于等待中,这是很糟糕的用户体验。可以加入进度条解决。注意到webview提供的两个方法:setWebViewClientsetWebChromeClient。其中setWebChromeClient方法正是可以处理progress的加载,此外,还可以处理js对话框,在webview中显示icon图标等。对于处理progress的代码片段如下:

[java]  view plain copy
  1. webView.setWebChromeClient(new WebChromeClient() {  
  2.     public void onProgressChanged(WebView view, int progress) {// 载入进度改变而触发  
  3.             if (progress == 100) {  
  4.                     handler.sendEmptyMessage(1);// 如果全部载入,隐藏进度对话框  
  5.             }  
  6.                 super.onProgressChanged(view, progress);  
  7.         }  
  8. });  

其中通过handler 消息机制来处理UI线程的更新:

 

[java]  view plain copy
  1. handler = new Handler() {  
  2.     public void handleMessage(Message msg) {// 定义一个Handler,用于处理下载线程与UI间通讯  
  3.         if (!Thread.currentThread().isInterrupted()){  
  4.             switch (msg.what) {  
  5.             case 0:  
  6.                 pd.show();// 显示进度对话框  
  7.                 break;  
  8.             case 1:  
  9.                 pd.hide();// 隐藏进度对话框,不可使用dismiss()、cancel(),否则再次调用show()时,显示的对话框小圆圈不会动。  
  10.                 break;  
  11.             }  
  12.         }  
  13.         super.handleMessage(msg);  
  14.     }  
  15. };  

对于setWebViewClient方法,一般用来处理html的加载(需要重载onPageStarted(WebView view, String url, Bitmap favicon))、关闭(需要重载onPageFinishedWebViewview, String url)方法)。

 

setWebViewClientsetWebChromeClient的作用:前者主要用于处理webView的控制问题,如加载、关闭、错误处理等;后者主要处理js对话框、图标、页面标题等。

4.        获取java中的数据

单独构建一个接口,作为处理jsjava的数据交互的桥梁,本文封装的代码AndroidToastForJs.java如下:

[java]  view plain copy
  1. public class AndroidToastForJs {  
  2.       
  3.     private Context mContext;  
  4.   
  5. public AndroidToastForJs(Context context){  
  6.         this.mContext = context;  
  7.     }  
  8.       
  9. //webview中调用toast原生组件  
  10. public void showToast(String toast) {  
  11.         Toast.makeText(mContext, toast, Toast.LENGTH_SHORT).show();  
  12.     }  
  13.       
  14. //webview中求和  
  15. public int sum(int a,int b){  
  16.         return a+b;  
  17.     }  
  18.       
  19.  //以json实现webview与js之间的数据交互  
  20. public String jsontohtml(){  
  21.         JSONObject map;  
  22.         JSONArray array = new JSONArray();  
  23.         try {  
  24.             map = new JSONObject();  
  25.             map.put("name","aaron");  
  26.             map.put("age"25);  
  27.             map.put("address""中国上海");  
  28.             array.put(map);  
  29.               
  30.             map = new JSONObject();  
  31.             map.put("name","jacky");  
  32.             map.put("age"22);  
  33.             map.put("address""中国北京");  
  34.             array.put(map);  
  35.               
  36.             map = new JSONObject();  
  37.             map.put("name","vans");  
  38.             map.put("age"26);  
  39.             map.put("address""中国深圳");  
  40.             map.put("phone","13888888888");  
  41.             array.put(map);  
  42.         } catch (JSONException e) {  
  43.             e.printStackTrace();  
  44.         }  
  45.         return array.toString();  
  46.     }  
  47. }  


 

Webview提供的传入js的方法:

[java]  view plain copy
  1. webView.addJavascriptInterface(new AndroidToastForJs(mContext), "JavaScriptInterface");  

Html页面jsonData.html设计的部分代码如下:

[html]  view plain copy
  1.     <script type="text/javascript">  
  2.     var result = JavaScriptInterface.jsontohtml();  
  3.     var obj = eval("("+result+")");//解析json字符串  
  4.     function showAndroidToast(toast)   
  5.     {          
  6.         JavaScriptInterface.showToast(toast);   
  7.     }  
  8.     function getjsonData(){  
  9.         var result = JavaScriptInterface.jsontohtml();  
  10.         var obj = eval("("+result+")");//解析json字符串  
  11.         for(i=0;i<obj.length;i++){  
  12.             var user=obj[i];  
  13.             document.write("<p>姓名:"+user.name+"</p>");  
  14.             document.write("<p>年龄:"+user.age+"</p>");  
  15.             document.write("<p>地址:"+user.address+"</p>");  
  16.             if(user.phone!=null){  
  17.                 document.write("<p>手机号码:"+user.address+"</p>");  
  18.             }  
  19.         }  
  20.     }     
  21.     function list(){  
  22.         document.write("<div data-role='header'><p>another</p></div>");  
  23.     }  
  24.     </script>  
  25. </head>   
  26. <body>   
  27. <div data-role="page" >  
  28.     <div data-role="header" data-theme="c">  
  29.         <h1>Android via Interface</h1>  
  30.     </div><!-- /header -->  
  31.     <div data-role="content">   
  32.         <button value="say hello" onclick="showAndroidToast('Hello,Android!')" data-theme="e"></button>  
  33.         <button value="get json data" onclick="getjsonData()" data-theme="e"></button>    
  34.     </div><!-- /content -->  
  35. <div data-role="collapsible" data-theme="c" data-content-theme="f">  
  36.    <h3>I'm <script>document.write(obj[0].name);</script>,click to see my info</h3>  
  37.    <p><script>document.write("<p>姓名:"+obj[0].name+"</p>");</script></p>  
  38.    <p><script>document.write("<p>年龄:"+obj[0].age+"</p>");</script></p>  
  39.    <p><script>document.write("<p>地址:"+obj[0].address+"</p>");</script></p>  
  40. </div>  
  41.     <div data-role="footer" data-theme="c">  
  42.         <h4>Page Footer</h4>  
  43.     </div><!-- /footer -->  
  44. </div><!-- /page -->  
  45. </body>  

点击say hello按钮运行的截图如下:


目录
相关文章
|
开发框架 前端开发 Android开发
Flutter 与原生模块(Android 和 iOS)之间的通信机制,包括方法调用、事件传递等,分析了通信的必要性、主要方式、数据传递、性能优化及错误处理,并通过实际案例展示了其应用效果,展望了未来的发展趋势
本文深入探讨了 Flutter 与原生模块(Android 和 iOS)之间的通信机制,包括方法调用、事件传递等,分析了通信的必要性、主要方式、数据传递、性能优化及错误处理,并通过实际案例展示了其应用效果,展望了未来的发展趋势。这对于实现高效的跨平台移动应用开发具有重要指导意义。
1339 4
|
6月前
|
安全 数据库 Android开发
在Android开发中实现两个Intent跳转及数据交换的方法
总结上述内容,在Android开发中,Intent不仅是活动跳转的桥梁,也是两个活动之间进行数据交换的媒介。运用Intent传递数据时需注意数据类型、传输大小限制以及安全性问题的处理,以确保应用的健壯性和安全性。
452 11
|
8月前
|
存储 XML Java
Android 文件数据储存之内部储存 + 外部储存
简介:本文详细介绍了Android内部存储与外部存储的使用方法及核心原理。内部存储位于手机内存中,默认私有,适合存储SharedPreferences、SQLite数据库等重要数据,应用卸载后数据会被清除。外部存储包括公共文件和私有文件,支持SD卡或内部不可移除存储,需申请权限访问。文章通过代码示例展示了如何保存、读取、追加、删除文件以及将图片保存到系统相册的操作,帮助开发者理解存储机制并实现相关功能。
2212 2
|
Java Linux Android开发
移动应用开发与操作系统的交互:深入理解Android和iOS
在数字时代,移动应用成为我们日常生活的一部分。本文将深入探讨移动应用开发的核心概念、移动操作系统的工作原理以及它们如何相互作用。我们将通过实际代码示例,展示如何在Android和iOS平台上创建一个简单的“Hello World”应用,并解释其背后的技术原理。无论你是初学者还是有经验的开发者,这篇文章都将为你提供有价值的见解和知识。
|
11月前
|
前端开发 Java Shell
【08】flutter完成屏幕适配-重建Android,增加GetX路由,屏幕适配,基础导航栏-多版本SDK以及gradle造成的关于fvm的使用(flutter version manage)-卓伊凡换人优雅草Alex-开发完整的社交APP-前端客户端开发+数据联调|以优雅草商业项目为例做开发-flutter开发-全流程-商业应用级实战开发-优雅草Alex
【08】flutter完成屏幕适配-重建Android,增加GetX路由,屏幕适配,基础导航栏-多版本SDK以及gradle造成的关于fvm的使用(flutter version manage)-卓伊凡换人优雅草Alex-开发完整的社交APP-前端客户端开发+数据联调|以优雅草商业项目为例做开发-flutter开发-全流程-商业应用级实战开发-优雅草Alex
779 20
【08】flutter完成屏幕适配-重建Android,增加GetX路由,屏幕适配,基础导航栏-多版本SDK以及gradle造成的关于fvm的使用(flutter version manage)-卓伊凡换人优雅草Alex-开发完整的社交APP-前端客户端开发+数据联调|以优雅草商业项目为例做开发-flutter开发-全流程-商业应用级实战开发-优雅草Alex
|
11月前
|
数据采集 JavaScript Android开发
【02】仿站技术之python技术,看完学会再也不用去购买收费工具了-本次找了小影-感觉页面很好看-本次是爬取vue需要用到Puppeteer库用node.js扒一个app下载落地页-包括安卓android下载(简单)-ios苹果plist下载(稍微麻烦一丢丢)-优雅草卓伊凡
【02】仿站技术之python技术,看完学会再也不用去购买收费工具了-本次找了小影-感觉页面很好看-本次是爬取vue需要用到Puppeteer库用node.js扒一个app下载落地页-包括安卓android下载(简单)-ios苹果plist下载(稍微麻烦一丢丢)-优雅草卓伊凡
523 7
【02】仿站技术之python技术,看完学会再也不用去购买收费工具了-本次找了小影-感觉页面很好看-本次是爬取vue需要用到Puppeteer库用node.js扒一个app下载落地页-包括安卓android下载(简单)-ios苹果plist下载(稍微麻烦一丢丢)-优雅草卓伊凡
|
数据采集 人工智能 自然语言处理
Midscene.js:AI 驱动的 UI 自动化测试框架,支持自然语言交互,生成可视化报告
Midscene.js 是一款基于 AI 技术的 UI 自动化测试框架,通过自然语言交互简化测试流程,支持动作执行、数据查询和页面断言,提供可视化报告,适用于多种应用场景。
3364 1
Midscene.js:AI 驱动的 UI 自动化测试框架,支持自然语言交互,生成可视化报告
|
存储 JavaScript 前端开发
【JavaScript】网页交互的灵魂舞者
本文介绍了 JavaScript 的三种引入方式(行内、内部、外部)和基础语法,包括变量、数据类型、运算符、数组、函数和对象等内容。同时,文章还详细讲解了 jQuery 的基本语法和常用方法,如 `text()`、`html()`、`val()`、`attr()` 和 `css()` 等,以及如何插入和删除元素。通过示例代码和图解,帮助读者更好地理解和应用这些知识。
179 1
【JavaScript】网页交互的灵魂舞者
|
缓存 JavaScript 前端开发
JavaScript 与 DOM 交互的基础及进阶技巧,涵盖 DOM 获取、修改、创建、删除元素的方法,事件处理,性能优化及与其他前端技术的结合,助你构建动态交互的网页应用
本文深入讲解了 JavaScript 与 DOM 交互的基础及进阶技巧,涵盖 DOM 获取、修改、创建、删除元素的方法,事件处理,性能优化及与其他前端技术的结合,助你构建动态交互的网页应用。
491 5
|
JSON 前端开发 JavaScript
聊聊 Go 语言中的 JSON 序列化与 js 前端交互类型失真问题
在Web开发中,后端与前端的数据交换常使用JSON格式,但JavaScript的数字类型仅能安全处理-2^53到2^53间的整数,超出此范围会导致精度丢失。本文通过Go语言的`encoding/json`包,介绍如何通过将大整数以字符串形式序列化和反序列化,有效解决这一问题,确保前后端数据交换的准确性。
376 4

热门文章

最新文章