android and webview详解手机开发

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

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

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

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

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

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

 1.  Java 调用 js 里面的函数,速度并不令人满意,大概一次一两百毫秒吧,如果要做交互性很强的事情,这种速度会让人疯掉的。而反过来就不一样了, js 去调 java 的方法,速度很快,基本上 40-50 毫秒一次。所以尽量用 js 调用 java 方法,而不是 java 去调用 js 函数。

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

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

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

五.        一些小问题。 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的数据重新拿回页面,效果图如下:

 android and 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;   
        }

 

 

 

原创文章,作者:奋斗,如若转载,请注明出处:https://blog.ytso.com/5995.html

(0)
上一篇 2021年7月17日
下一篇 2021年7月17日

相关推荐

发表回复

登录后才能评论