android中webview使用的一些细节

转自: http://www.cnblogs.com/mingforyou/archive/2012/12/06/2805182.html

webview ,用网页来布局。 Android 的 webview 是基于 webkit 内核,不过他的运行效果和 firefox 上如出一辙,因此写的时候都是先用 firefox 测试,测试 OK 了再放到程序里面看效果,基本上不会有什么问题。其实 android 的 webview 跟 iphone 的 webview 差很少, iphone 上的 webview 比 android 上的强大多了。 javascript

谈一下研究 webview 的一些成果: css

一.    加载资源的速度不慢,可是资源多了,就很慢。图片、 css 、 js 、 html 这些资源每一个大概须要 10-200ms ,通常都是 30ms 就 ok 了。若是一个页面上的资源不少,就很浪费时间。 html

二.    Js 和 css 的执行速度。开始的时候,个人页面都是用 js 生成 DOM ,添加样式等也用 js 添加。后来发现,加载一个页面竟然要 5-6 秒。而后我就怀疑是否是 js 的执行效率不高,而后就把能用 css 的地方都用 css ,能直接写到 html 上的就不用 js 动态生成。结果,速度并无多大的提高,最多提高了 1 秒。看来, Js 的执行速度虽然比不上 css ,可是还不至于慢到那种程度。那会是什么缘由使得页面加载速度这么慢?通过仔细的排查,最终发现,是由于我用了 jQuery 框架。 java

Webview 加载页面的顺序是这样的:先加载 html ,而后从里面解析出 css 、 js 文件和页面上写死的图片资源进行加载,若是 webkit 的缓存里面有,就不加载。加载完这些资源以后,就进行 css 的渲染和 js 的执行。 Css 的渲染通常不须要很长时间,几十毫秒就 ok 。关键是 js 的执行,若是用了 jQuery ,则执行起来须要 5-6 秒。而在这段时间,若是不在 webview 里设置背景,网页部分是白色的,很难看。这是一个很糟糕的用户体验。因此若是用网页布局程序,最好别用很大的 js 框架。 jquery

三.     网页和 Java 之间的互调。这个功能是 iphone 里面就有的,网上也有不少资料,能够告诉咱们怎么作,这些都是很简单、很基本的。我研究了一段时间,总结一下: android

 1.  Java 调用 js 里面的函数,速度并不使人满意,大概一次一两百毫秒吧,若是要作交互性很强的事情,这种速度会让人疯掉的。而反过来就不同了, js 去调 java 的方法,速度很快,基本上 40-50 毫秒一次。因此尽可能用 js 调用 java 方法,而不是 java 去调用 js 函数。 ios

 2.  Java 调用 js 的函数,没有返回值,而 Js 调用 java 方法,能够有返回值。返回值能够是字符串,也能够是对象。若是是字符串,有个很讨厌的问题,第 3 点我会讲的。若是是对象,这个对象会被转换为 js 的对象,直接能够访问里面的方法。可是我不推荐 java 返回给 js 的是对象,除非是必须。由于 js 收到 java 返回的对象,会产生一些交换对象,而若是这些对象的数量增长到了 500 或 600 以上,程序就会出问题。因此尽可能返回基本数据类型或者字符串。 web

3.  Js 调用 java 的方法,返回值若是是字符串,你会发现这个字符串是 native 的,不能对它进行一些修改操做,好比想对它 substr ,取不到。怎么解决呢?转成 locale 的。使用 toLocaleString() 函数就能够了。不过这个函数的速度并不快,转化的字符串若是不少,将会很耗费时间。 json

四.        网页上拖动元素。网页上有一个 div ,想要拖动它到另一个地方,怎么作?若是用 PC 上的网页作法,监听 onmousedown 、 onmousemove 和 onmouseup 就能够了。可是在手机上,事件模型就不同了。在网页上点击,拖动,而后释放,手离开屏幕的时候, webview 才会触发 onmousedown 、 onmousemove 、 onmouseup 事件。因此,要想拖动,不能这么作。这个问题困扰我很长时间,后来发现 iphone 上的作法,才解决了。 Iphone 上的 webview 有专为触摸屏设计的事件 ontouchstart 、 ontouchmove 、 ontouchend ,这几个事件的响应是实时的,就能解决拖动的问题了。 windows

五.        一些小问题。 Webview 里面的网页,若是有 input ,须要输入,可是点上去却没反应,输入法不出来。这种状况是由于 webview 没有获取焦点。须要在 java 里面给 webview 设置一下 requestFocus() 就好了。

六.        Android 上的 webview 和 iphone 的 webview 区别。目前为止,我发现的区别有这么几个:

1 . Android 上, webview 不支持多点触控,没有 ongesture 系列事件,而 iphone 上有。

 2 . Android 上的 webview 不支持透明, iphone 上能够。

 

浏览器控件是每一个开发环境都具有的,这为马甲神功提供了用武之地,windows的有webbrowserandroidios都有webview。只是其引擎不一样,相对于微软的webbrowserandroidioswebview的引擎都是webkit,对Html5提供支持。本篇主要介绍androidwebview之强大。

A.    webview组件如何使用

 

1)       添加权限:AndroidManifest.xml中必须使用许可"android.permission.INTERNET",不然会出Web page not available错误。

2)       在要Activity中生成一个WebView组件:WebView webView = new WebView(this);或者能够在activitylayout文件里添加webview控件:

复制代码
<WebView android:id="@+id/wv" android:layout_width="fill_parent" android:layout_height="fill_parent" android:text="@string/hello" />
复制代码

 

3)       设置WebView基本信息:
          若是访问的页面中有Javascript,则webview必须设置支持Javascript
          webview.getSettings().setJavaScriptEnabled(true);  
          触摸焦点起做用
          requestFocus();
          取消滚动条
          this.setScrollBarStyle(SCROLLBARS_OUTSIDE_OVERLAY);

 

4)       设置WevView要显示的网页:
          互联网用:webView.loadUrl("http://www.google.com"); 
          本地文件用:webView.loadUrl("file:///android_asset/XX.html");  本地文件存放在:assets文件中

5)       若是但愿点击连接由本身处理,而不是新开Android的系统browser中响应该连接。给WebView添加一个事件监听对象(WebViewClient)并重写其中的一些方法:
 shouldOverrideUrlLoading:对网页中超连接按钮的响应。当按下某个链接时WebViewClient会调用这个方法,并传递参数:按下的url。好比当webview内嵌网页的某个数字被点击时,它会自动认为这是一个电话请求,会传递urltel:123,若是你不但愿如此可经过重写shouldOverrideUrlLoading函数解决:

  

 
复制代码
public boolean shouldOverrideUrlLoading(WebView view,String url){ if(url.indexOf("tel:")<0){//页面上有数字会致使链接电话  view.loadUrl(url); } return true; }
复制代码

 



          另外还有其余一些可重写的方法  
1
,接收到 Http 请求的事件
onReceivedHttpAuthRequest(WebView view, HttpAuthHandler handler, String host, String realm) 

 

2,打开连接前的事件
public boolean shouldOverrideUrlLoading(WebView view, String url) { view.loadUrl(url); return true; } 

这个函数咱们能够作不少操做,好比咱们读取到某些特殊的URL,因而就能够不打开地址,取消这个操做,进行预先定义的其余操做,这对一个程序是很是必要的。
 
3
,载入页面完成的事件
public void onPageFinished(WebView view, String url){ } 

一样道理,咱们知道一个页面载入完成,因而咱们能够关闭loading条,切换程序动做。
 
4
,载入页面开始的事件
public void onPageStarted(WebView view, String url, Bitmap favicon) { } 

这个事件就是开始载入页面调用的,一般咱们能够在这设定一个loading的页面,告诉用户程序在等待网络响应。
 
经过这几个事件,咱们能够很轻松的控制程序操做,一边用着浏览器显示内容,一边监控着用户操做实现咱们须要的各类显示方式,同时能够防止用户产生误操做。
      

6)       若是用webview点连接看了不少页之后,若是不作任何处理,点击系统“Back”键,整个浏览器会调用finish()而结束自身,若是但愿浏览的网页回退而不是退出浏览器,须要在当前Activity中处理并消费掉该Back事件。
      
 覆盖Activity类的onKeyDown(int keyCoder,KeyEvent event)方法。

  

复制代码
public boolean onKeyDown(int keyCoder,KeyEvent event){ if(webView.canGoBack() && keyCoder == KeyEvent.KEYCODE_BACK){ webview.goBack(); //goBack()表示返回webView的上一页面 return true; } return false; }
复制代码

 



B.    Webviewjs交互

Webviewjs的双向交互才是androidwebview强大所在,也是马甲精神可以完全执行的基础保障。

首先,webview能够定义一个在其内嵌页面中能够触发的事件

 

复制代码
wv.addJavascriptInterface(new DemoJavaScriptInterface(), "demo"); private final class DemoJavaScriptInterface { DemoJavaScriptInterface(){} public void clickonAndroid( final String order){ mHandler.post(newRunnable(){ @Override public void run(){ jsonText="{"name":""+order+""}"; wv.loadUrl("javascript:wave("+jsonText+")"); } }); } }
复制代码

 

 

经过以上代码,便可实如今其内嵌网页中触发window.demo.clickOnAndroid(str)事件并传参数strwebviewWebview接收到str以后,能够经过以上代码触发其内嵌页面中的js函数wave(str)。这样就能够实现网页触发webview的事件并传参数,webview接收参数并调用js函数。

下面看个人Html脚本:

复制代码
<!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN""http://www.w3.org/TR/html4/loose.dtd"> <html> <head> <meta http-equiv="Content-Type" content="text/html; charset=UTF-8"> <title>Insert title here</title> <script type="text/javascript" src="jquery.js"></script> <script> function toclient() { var order=$("#val").val(); window.demo.clickonAndroid(order); } function wave(str){ //alert(str.name);  $("#fromclient").text(str.name); } </script> </head> <body>这是一个html页面 <br/> 输入一个字符串:<br/> <input id="val" /> <input type="submit" value="点击提交给客户端" onclick="toclient();"/> <br /> 显示返回:<label id="fromclient"></label> </body> </html>
复制代码

 

 

经过脚本看到wavestr)函数是负责将原来传给webview的数据从新拿回页面,效果图以下:

 

 

另外,若是你想获取页面的一些处理数据并交给webview客户端处理,可在wave函数里将数据alert,而后webview中重写WebChromeClientonJsAlert函数,具体代码以下

复制代码
wv.setWebChromeClient(new MyWebChromeClient()); final class MyWebChromeClient extends WebChromeClient{ @Override public booleanonJsAlert(WebView view, String url, String message, final JsResult result) { //message就是wave函数里alert的字符串,这样你就能够在android客户端里对这个数据进行处理  result.confirm(); } return true; }
复制代码
相关文章
相关标签/搜索