WebView库功能完善
目录介绍
- 01.loadUrl到底做了什么
- 02.触发加载网页的行为
- 03.webView重定向怎么办
- 04.js交互的一点知识分享
- 05.拦截缓存如何优雅处理
- 06.关于一些问题和优化
- 07.关于一点面向对象思想
- 08.关于后期需要研究目标
01.loadUrl到底做了什么
- WebView.loadUrl(url)加载网页做了什么?
- 加载网页是一个复杂的过程,在这个过程中,我们可能需要执行一些操作,包括:
- 加载网页前,重置WebView状态以及与业务绑定的变量状态。WebView状态包括重定向状态(mTouchByUser)、前端控制的回退栈(mBackStep)等,业务状态包括进度条、当前页的分享内容、分享按钮的显示隐藏等。
- 加载网页前,根据不同的域拼接本地客户端的参数,包括基本的机型信息、版本信息、登录信息以及埋点使用的Refer信息等,有时候涉及交易、财产等还需要做额外的配置。
- 开始执行页面加载操作时,会回调WebViewClient.onPageStarted(webView,url,favicon)。在此方法中,可以重置重定向保护的变量(mRedirectProtected),当然也可以在页面加载前重置,由于历史遗留代码问题,此处尚未省去优化。
- 加载页面的过程中回调哪些方法?
- WebChromeClient.onReceivedTitle(webview, title),用来设置标题。需要注意的是,在部分Android系统版本中可能会回调多次这个方法,而且有时候回调的title是一个url,客户端可以针对这种情况进行特殊处理,避免在标题栏显示不必要的链接。
- WebChromeClient.onProgressChanged(webview, progress),根据这个回调,可以控制进度条的进度(包括显示与隐藏)。一般情况下,想要达到100%的进度需要的时间较长(特别是首次加载),用户长时间等待进度条不消失必定会感到焦虑,影响体验。其实当progress达到80的时候,加载出来的页面已经基本可用了。事实上,国内厂商大部分都会提前隐藏进度条,让用户以为网页加载很快。
- WebViewClient.shouldInterceptRequest(webview, request),无论是普通的页面请求(使用GET/POST),还是页面中的异步请求,或者页面中的资源请求,都会回调这个方法,给开发一次拦截请求的机会。在这个方法中,我们可以进行静态资源的拦截并使用缓存数据代替,也可以拦截页面,使用自己的网络框架来请求数据。包括后面介绍的WebView免流方案,也和此方法有关。
- WebViewClient.shouldOverrideUrlLoading(webview, request),如果遇到了重定向,或者点击了页面中的a标签实现页面跳转,那么会回调这个方法。可以说这个是WebView里面最重要的回调之一,后面WebView与Native页面交互一节将会详细介绍这个方法。
- WebViewClient.onReceivedError(webview,handler,error),加载页面的过程中发生了错误,会回调这个方法。主要是http错误以及ssl错误。在这两个回调中,我们可以进行异常上报,监控异常页面、过期页面,及时反馈给运营或前端修改。在处理ssl错误时,遇到不信任的证书可以进行特殊处理,例如对域名进行判断,针对自己公司的域名“放行”,防止进入丑陋的错误证书页面。也可以与Chrome一样,弹出ssl证书疑问弹窗,给用户选择的余地。
- 加载页面结束回调哪些方法
- 会回调WebViewClient.onPageFinished(webview,url)。
- 这时候可以根据回退栈的情况判断是否显示关闭WebView按钮。通过mActivityWeb.canGoBackOrForward(-1)判断是否可以回退。
02.触发加载网页的行为
- 触发加载网页的行为主要有两种方式:
- (A)点击页面,触发<link>标签。
- (B)调用WebView的loadUrl()方法
- 这两种方法都会发出一条地址,区别就在于这条地址是目的地址还是重定向地址。以访问http://www.baidu.com百度的页面来测试一下方法的执行顺序。
- 触发加载网页流程分析
- 在代码中通过loadUrl加载百度的首页,此时的行为属于(B)方式。
- 可以发现大概的执行顺序是:onPageStarted ——> shouldOverrideUrlLoading ——> onPageFinished
- 那么为什么会执行多次呢,思考一下?具体可以看一下7.2得出的结论分析。
X5LogUtils: -------onPageStarted-------http://www.baidu.com/
X5LogUtils: -------shouldOverrideUrlLoading-------https://m.baidu.com/?from=844b&vit=fps
X5LogUtils: -------onPageFinished-------http://www.baidu.com/
X5LogUtils: -------onPageStarted-------https://m.baidu.com/?from=844b&vit=fps
X5LogUtils: -------onReceivedTitle-------百度一下
X5LogUtils: -------shouldOverrideUrlLoading-------http://m.baidu.com/?cip=117.101.19.67&baiduid=C6FCEED198C994E0D653C094F2708C32&from=844b&vit=fps?from=844b&vit=fps&index=&ssid=0&bd_page_type=1&logid=12175252243175665635&pu=sz%401321_480&t_noscript=jump
X5LogUtils: -------onPageFinished-------https://m.baidu.com/?from=844b&vit=fps
X5LogUtils: -------shouldOverrideUrlLoading-------https://m.baidu.com/?cip=117.101.19.67&baiduid=C6FCEED198C994E0D653C094F2708C32&from=844b&vit=fps?from=844b&vit=fps&index=&ssid=0&bd_page_type=1&logid=12175252243175665635&pu=sz%401321_480&t_noscript=jump
X5LogUtils: -------onPageStarted-------http://m.baidu.com/?cip=117.101.19.67&baiduid=C6FCEED198C994E0D653C094F2708C32&from=844b&vit=fps?from=844b&vit=fps&index=&ssid=0&bd_page_type=1&logid=12175252243175665635&pu=sz%401321_480&t_noscript=jump
X5LogUtils: -------onPageFinished-------http://m.baidu.com/?cip=117.101.19.67&baiduid=C6FCEED198C994E0D653C094F2708C32&from=844b&vit=fps?from=844b&vit=fps&index=&ssid=0&bd_page_type=1&logid=12175252243175665635&pu=sz%401321_480&t_noscript=jump
X5LogUtils: -------onPageStarted-------https://m.baidu.com/?cip=117.101.19.67&baiduid=C6FCEED198C994E0D653C094F2708C32&from=844b&vit=fps?from=844b&vit=fps&index=&ssid=0&bd_page_type=1&logid=12175252243175665635&pu=sz%401321_480&t_noscript=jump
X5LogUtils: -------onReceivedTitle-------百度一下,你就知道
X5LogUtils: -------onPageFinished-------https://m.baidu.com/?cip=117.101.19.67&baiduid=C6FCEED198C994E0D653C094F2708C32&from=844b&vit=fps?from=844b&vit=fps&index=&ssid=0&bd_page_type=1&logid=12175252243175665635&pu=sz%401321_480&t_noscript=jump
- 在首页,点击一下“hao123”,跳转到www.hao123.com的主页上来,此时的行为属于(A)方式。
- 可以发现大概的执行顺序是:shouldOverrideUrlLoading ——> onPageStarted ——> onPageFinished
X5LogUtils: -------shouldOverrideUrlLoading-------http://m.hao123.com/?ssid=0&from=844b&bd_page_type=1&uid=0&pu=sz%401321_1002%2Cta%40utouch_2_9.0_2_6.2&idx=30000&itj=39
X5LogUtils: -------onPageStarted-------http://m.hao123.com/?ssid=0&from=844b&bd_page_type=1&uid=0&pu=sz%401321_1002%2Cta%40utouch_2_9.0_2_6.2&idx=30000&itj=39
X5LogUtils: -------onReceivedTitle-------hao123导航-上网从这里开始
X5LogUtils: -------onPageFinished-------http://m.hao123.com/?ssid=0&from=844b&bd_page_type=1&uid=0&pu=sz%401321_1002%2Cta%40utouch_2_9.0_2_6.2&idx=30000&itj=39
- 然后在hao123页面,点击优酷网进行跳转,此时的行为属于(A)方式。
X5LogUtils: -------shouldOverrideUrlLoading-------http://m.hao123.com/j.php?z=2&page=index_cxv3&pos=cydhwt_n2&category=ty&title=%E4%BC%98%E9%85%B7%E7%BD%91&qt=tz&url=http%3A%2F%2Fwww.youku.com%2F&key=58193753e7a868d9a013056c6c4cd77b
X5LogUtils: -------onPageStarted-------http://m.hao123.com/j.php?z=2&page=index_cxv3&pos=cydhwt_n2&category=ty&title=%E4%BC%98%E9%85%B7%E7%BD%91&qt=tz&url=http%3A%2F%2Fwww.youku.com%2F&key=58193753e7a868d9a013056c6c4cd77b
X5LogUtils: -------shouldOverrideUrlLoading-------http://www.youku.com/
X5LogUtils: -------onPageFinished-------http://m.hao123.com/j.php?z=2&page=index_cxv3&pos=cydhwt_n2&category=ty&title=%E4%BC%98%E9%85%B7%E7%BD%91&qt=tz&url=http%3A%2F%2Fwww.youku.com%2F&key=58193753e7a868d9a013056c6c4cd77b
X5LogUtils: -------onPageStarted-------http://www.youku.com/
X5LogUtils: -------shouldOverrideUrlLoading-------https://www.youku.com/
X5LogUtils: -------onPageFinished-------http://www.youku.com/
X5LogUtils: -------onPageStarted-------https://www.youku.com/
X5LogUtils: -------onReceivedTitle-------优酷视频-首页
X5LogUtils: -------onPageFinished-------https://www.youku.com/
- 然后从优酷页面回退到hao123页面,看看又回执行哪些方法。
X5LogUtils: -------onPageStarted-------http://m.hao123.com/?ssid=0&from=844b&bd_page_type=1&uid=0&pu=sz%401321_1002%2Cta%40utouch_2_9.0_2_6.2&idx=30000&itj=39
X5LogUtils: -------onReceivedTitle-------hao123导航-上网从这里开始
X5LogUtils: -------onReceivedTitle-------hao123导航-上网从这里开始
X5LogUtils: -------onPageFinished-------http://m.hao123.com/?ssid=0&from=844b&bd_page_type=1&uid=0&pu=sz%401321_1002%2Cta%40utouch_2_9.0_2_6.2&idx=30000&itj=39
- 然后从hao123页面回退到百度首页,看看又回执行哪些方法。
X5LogUtils: -------onPageStarted-------https://m.baidu.com/?cip=117.101.19.67&baiduid=C6FCEED198C994E0D653C094F2708C32&from=844b&vit=fps?from=844b&vit=fps&index=&ssid=0&bd_page_type=1&logid=12175252243175665635&pu=sz%401321_480&t_noscript=jump
X5LogUtils: -------onReceivedTitle-------百度一下,你就知道
X5LogUtils: -------onReceivedTitle-------百度一下,你就知道
X5LogUtils: -------onPageFinished-------https://m.baidu.com/?cip=117.101.19.67&baiduid=C6FCEED198C994E0D653C094F2708C32&from=844b&vit=fps?from=844b&vit=fps&index=&ssid=0&bd_page_type=1&logid=12175252243175665635&pu=sz%401321_480&t_noscript=jump
- 在代码中通过loadUrl加载百度的首页,此时的行为属于(B)方式。
- 得出结论分析说明
- 在(A)行为方式下(用户点击链接的回调):
- 1.如果是目的地址,那么方法的执行顺序是:
- shouldOverrideUrlLoading() -> onPageStarted()-> onPageFinished()
- shouldOverrideUrlLoading()由于它要提供给APP选择加载网页环境的机会,所以只要是网页上地址请求,都会获取到。
- 2.如果是重定向地址,在跳转到目的地址之前会进行不断的地址定位,每一次地址定位都会由以下执行顺序体现出来:
- onPageStarted()->shouldOverrideUrlLoading()->onPageFinished()
- 暂且设定这种执行顺序叫:fixed position
- 那么一个正常的重定向地址,方法的执行顺序就是:
- shouldOverrideUrlLoading() -> fixed position -> … -> fixed position -> onPageStarted() -> onPageFinished()
- 举个例子:有重定向(A->B->C),那么
shouldOverrideUrlLoading(A) -> onPageStarted(A) ->
onPageStarted(B) -> shouldOverrideUrlLoading(B) ->
onPageStarted(C) -> shouldOverrideUrlLoading(C) -> onPageFinished(C)
- 1.如果是目的地址,那么方法的执行顺序是:
- 在(B)行为下:
- 1.如果是目的地址,那么方法的执行顺序是:
- onPageStarted()-> onPageFinished()
- loadUrl()加载地址时,一般不会触发shouldOverrideUrlLoading(),一旦触发了,就说明这是一个重定向地址。
- 2.如果是重定向地址,方法的执行顺序就是:
- fixed position -> … -> fixed position -> onPageStarted() -> onPageFinished()
- 1.如果是目的地址,那么方法的执行顺序是:
- 在(A)行为方式下(用户点击链接的回调):
03.webView重定向怎么办
- webView出现302/303重定向
- 302重定向又称之为302代表暂时性转移,比如你跳转A页面,但由于网页添加了约束条件,可能让你跳转到B页面,甚至多次重定向。
- 导致的问题
- 1.A-->B-->C,比如你跳转A页面,最终重定向到C页面。这个时候调用goBack方法,返回到B链接,但是B链接又会跳转到C链接,从而导致没法返回到A链接界面
- 2.会多次执行onPageStarted和onPageFinished,如果你这里有加载进度条或者loading,那么会导致进度条或者loading执行多次
- 常见的解决方案
- 手动管理回退栈,遇到重定向时回退两次。
- 通过HitTestResult判断是否是重定向,从而决定是否自己加载url。具体看:16.301/302回退栈问题解决方案2
- 通过设置标记位,在onPageStarted和onPageFinished分别标记变量避免重定向。具体看:17.301/302回退栈问题解决方案3
- 通过用户的touch事件来判断重定向。具体看:15.301/302回退栈如何处理1
- 如何判断重定向
- 通过getHitTestResult()返回值,如果返回null,或者UNKNOWN_TYPE,则表示为重定向。具体看:18.如何用代码判断是否重定向
- 在加载一个页面开始的时候会回调onPageStarted方法,在该页面加载完成之后会回调onPageFinished方法。而如果该链接发生了重定向,回调shouldOverrideUrlLoading会在回调onPageFinished之前。
- 终极解决方案如下
- 需要准备的条件
- 创建一个栈,主要是用来存取和移除url的操作。这个url包括所有的请求链接
- 定义一个变量,用于判断页面是否处于正在加载中。
- 定义一个变量,用于记录重定向前的链接url
- 定一个重定向时间间隔,主要为了避免刷新造成循环重定向
- 具体怎么操作呢
- 在执行onPageStarted时,先移除栈中上一个url,然后将url加载到栈中。
- 当出现错误重定向的时候,如果和上一次重定向的时间间隔大于3秒,则reload页面。
- 在回退操作的时候,判断如果可以回退,则从栈中获取最后停留的url,然后loadUrl。即可解决回退问题。
- 具体方法思路
- 可以看:20.重定向终极优雅解决方案
- 具体代码看:X5WebViewClient
- 需要准备的条件
04.js交互的一点知识分享
- js交互介绍
- Java调用js方法有两种:
- WebView.loadUrl("javascript:" + javascript);
- WebView.evaluateJavascript(javascript, callbacck);
- js调用Java的方法有三种,分别是:
- JavascriptInterface
- WebViewClient.shouldOverrideUrlLoading()
- WebChromeClient.onJsPrompt()
- Java调用js方法有两种:
- js调用java方法比较和区别分析
- 1.通过 addJavascriptInterface 方法进行添加对象映射。js最终通过对象调用原生方法
- 2.shouldOverrideUrlLoading拦截操作,获取scheme匹配,与网页约定好一个协议,如果匹配,执行相应操作
- 3.利用WebChromeClient回调接口onJsPrompt拦截操作。
- onJsAlert 是不能返回值的,而 onJsConfirm 只能够返回确定或者取消两个值,只有 onJsPrompt 方法是可以返回字符串类型的值,操作最全面方便。
- 详细分析可以看:03.Js调用Android
- js调用java原生方法可能存在的问题?
- 提出问题
- 1.原生方法是否可以执行耗时操作,如果有会阻塞通信吗?4.4.8 prompt的一个坑导致js挂掉
- 2.多线程中调用多个原生方法,如何保证原生方法每一个都会被执行到?
- 3.js会阻塞等待当前原生函数(耗时操作的那个)执行完毕再往下走,所以js调用java方法里面最好也不要做耗时操作
- 解决方案
- 1.在js调用window.alert,window.confirm,window.prompt时,会调用WebChromeClient对应方法,可以此为入口,作为消息传递通道,考虑到开发习惯,一般不会选择alert跟confirm,通常会选prompt作为入口,在App中就是onJsPrompt作为jsbridge的调用入口。由于onJsPrompt是在UI线程执行,所以尽量不要做耗时操作,可以借助Handler灵活处理。
- 2.利用Handler封装一下,让每个任务自己处理,耗时的话就开线程自己处理。具体可以看:WvWebView
- 提出问题
- java调用js的时机
- onPageFinished()或者onPageStarted()方法中注入js代码吗?
- js交互,大部分都会认为js在WebViewClient.onPageFinished()方法中注入最合适,此时dom树已经构建完成,页面已经完全展现出来。但如果做过页面加载速度的测试,会发现WebViewClient.onPageFinished()方法通常需要等待很久才会回调(首次加载通常超过3s),这是因为WebView需要加载完一个网页里主文档和所有的资源才会回调这个方法。
- 能不能在WebViewClient.onPageStarted()中注入呢?答案是不确定。经过测试,有些机型可以,有些机型不行。在WebViewClient.onPageStarted()中注入还有一个致命的问题——这个方法可能会回调多次,会造成js代码的多次注入。
- 从7.0开始,WebView加载js方式发生了一些小改变,官方建议把js注入的时机放在页面开始加载之后。
- 可以在onProgressChanged中方法中注入js代码
- 页面的进度加载到80%的时候,实际上dom树已经渲染得差不多了,表明WebView已经解析了标签,这时候注入一般是成功的。
- 提到的多次注入控制,使用了boolean值变量控制;重新加载一个URL之前,需要重置boolean值变量,让重新加载后的页面再次注入js
- onPageFinished()或者onPageStarted()方法中注入js代码吗?
05.拦截缓存如何优雅处理
- WebView为何加载慢
- webView是怎么加载网页的呢?
- webView初始化->DOM下载→DOM解析→CSS请求+下载→CSS解析→渲染→绘制→合成
- 渲染速度慢
- 前端H5页面渲染的速度取决于 两个方面:
- Js 解析效率。Js 本身的解析过程复杂、解析速度不快 & 前端页面涉及较多 JS 代码文件,所以叠加起来会导致 Js 解析效率非常低
- 手机硬件设备的性能。由于Android机型碎片化,这导致手机硬件设备的性能不可控,而大多数的Android手机硬件设备无法达到很好很好的硬件性能
- 前端H5页面渲染的速度取决于 两个方面:
- 页面资源加载缓慢
- H5 页面从服务器获得,并存储在 Android手机内存里:
- H5页面一般会比较多
- 每加载一个 H5页面,都会产生较多网络请求:
- HTML 主 URL 自身的请求;
- HTML外部引用的JS、CSS、字体文件,图片也是一个独立的 HTTP 请求
- 每一个请求都串行的,这么多请求串起来,这导致 H5页面资源加载缓慢
- H5 页面从服务器获得,并存储在 Android手机内存里:
- webView是怎么加载网页的呢?
- 解决WebView加载慢
- 前端H5的缓存机制(WebView 自带)
- 资源拦截缓存
- 资源拦截替换
- webView浏览器缓存机制
- 这些技术都是协议层所定义的,在Android的webView当中我们可以通过配置决定是否采纳这几个协议的头部属性
// LOAD_CACHE_ONLY: 不使用网络,只读取本地缓存数据
// LOAD_DEFAULT: (默认)根据cache-control决定是否从网络上取数据。
// LOAD_NO_CACHE: 不使用缓存,只从网络获取数据.
// LOAD_CACHE_ELSE_NETWORK,只要本地有,无论是否过期,或者no-cache,都使用缓存中的数据。
ws.setCacheMode(WebSettings.LOAD_DEFAULT);
- 一般设置为默认的缓存模式就可以了。关于缓存的配置, 主要还是靠web前端和后台设置。关于浏览器缓存机制
- 自身构建缓存方案
- 拦截处理
- 在shouldInterceptRequest方法中拦截处理
- 步骤1:判断拦截资源的条件,即判断url里的图片资源的文件名
- 步骤2:创建一个输入流,这里可以先从内存中拿,拿不到从磁盘中拿,再拿不到就从网络获取数据
- 步骤3:打开需要替换的资源(存放在assets文件夹里),或者从lru中取出缓存的数据
- 步骤4:替换资源
- 有几个问题
- 如何判断url中资源是否需要拦截,或者说是否需要缓存
- 如何缓存js,css等
- 缓存数据是否有时效性
- 关于缓存下载的问题,是引入okhttp还是原生网络请求,缓存下载失败该怎么处理
- 在哪里进行拦截
- webView在加载网页的时候,用户能够通过系统提供的API干预各个中间过程。我们要拦截的就是网页资源请求的环节。这个过程,WebViewClient当中提供了以下两个入口:
// android5.0以上的版本加入
@Override
public WebResourceResponse shouldInterceptRequest(WebView webView, WebResourceRequest webResourceRequest) {
return super.shouldInterceptRequest(webView, webResourceRequest);
} @Override
public WebResourceResponse shouldInterceptRequest(WebView webView, String s) {
return super.shouldInterceptRequest(webView, s);
}
- 替换资源操作
- 只要在这两个入口构造正确的WebResourceResponse对象,就可以替换默认的请求为我们提供的资源
- 因此,在每次请求资源的时候根据请求的URL/WebResourceRequest判断是否存在本地的缓存,并在缓存存在的情况下将缓存的输入流返回
- webView在加载网页的时候,用户能够通过系统提供的API干预各个中间过程。我们要拦截的就是网页资源请求的环节。这个过程,WebViewClient当中提供了以下两个入口:
- 拦截处理
06.关于一些问题和优化
- 影响页面加载的一些因素有那些?
- 1.加载网页中,如果图片很多,而这些图片的请求又是一个个独立并且串行的请求。那么可能会导致加载页面比较缓慢……
- 2.app原生和webView中请求,都会涉及到https的网络请求,那么在请求前会有域名dns的解析,这个也会有大约200毫秒的解析时间(主要耗费时间dns,connection,服务器处理等)……
- 3.webView加载html网页时,有些js一直在执行比如动画之类的东西,此刻webView挂在了后台这些资源是不会被释放用户也无法感知。导致耗费资源……
- 4.关于加载loading或者加载进度条,不一定要放到onPageStarted开始执行即显示出来,因为webView从创建到这个方法会有一个时间……
- 5.webView默认开启密码保存功能,如果网页涉及到用户登陆,密码会被明文保到 /data/data/com.package.name/databases/webview.db 中,这样就有被盗取密码的危险……
- 6.h5页面被拦截或者注入广告,重定向,或者DNS劫持。一般跟连接的wifi有关系(http劫持),也可能跟运营商有关系(dns劫持)
- 具体可以操作的优化分析
- 1.加载webView中的资源时,针对图片,等页面finish后再发起图片加载(也就是执行onPageFinished设置加载图片)。具体看5.0.2图片加载次序优化
- 2.DNS域名解析采用和客户端API相同的域名, DNS会在系统级别进行缓存,对于WebView的地址,如果使用的域名与native的API相同,则可以直接使用缓存的DNS而不用再发起请求图片。具体看5.0.7 DNS采用和客户端API相同的域名
- 3.在后台的时候,会调用onStop方法,即此时关闭js交互,回到前台调用onResume再开启js交互。具体看5.0.9 后台无法释放js导致发热耗电
- 4.提前显示进度条不是提升性能,但是对用户体验来说也是很重要的一点 ,WebView.loadUrl("url") 不会立马就回调onPageStarted方法,因为在这一时间段,WebView 有可能在初始化内核,也有可能在与服务器建立连接,这个时间段容易出现白屏
- 5.需要通过 WebSettings.setSavePassword(false) 关闭密码保存功能。
- 6.一般可以处理:1使用https代替http;2.添加白名单(比如添加自己网站的host,其他不给访问);3.对页面md5校验(不太好)。设置白名单参考:5.0.8 如何设置白名单操作
- 还有一些其他的优化小细节
- a.WebView处理404、500逻辑,在WebChromeClient子类中可以重写他的onReceivedTitle()方法监听标题,还有在WebChromeClient子类中onReceivedHttpError可以监听statusCode。具体操作看5.1.5 WebView处理404、500逻辑
- b.如果不显示图片,开发的时候可能使用的是https的链接, 但是链接中的图片可能是http的,需要开启设置。具体看:4.1.4 webView加载网页不显示图片
- c.evaluateJavascript(String var1, ValueCallback<String> var2)中url长度有限制,在19以上超过2097152个字符失效,这个地方可以加个判断。不过一般很难碰到……具体可以参考:4.3.1 Android与js传递数据大小有限制
- d.在web页面android软键盘覆盖问题,常见的有android:windowSoftInputMode的值adjustPan或者adjustResize即可,如果webView是全屏模式则仍然会出现问题。具体看:4.6.1 在web页面android软键盘覆盖问题
- e.关于WebView隐藏H5页面中的某个标签视图,大概操作就是在页面加载完成,通过getElementsByClassName找到h5中标签name,然后手动写function方法隐藏标签。但加载时机很关键,不过会造成闪屏和多次加载。具体看:4.6.6 WebView如何隐藏H5的部分内容问题
- f.页面重定向,会导致onPageStarted多次执行,那么这个时候如何避免加载进度条出现执行多次,或者跳动的问题。具体可见:09.web进度条避免多次加载
- g.建议开启Google安全浏览服务,用户访问不安全网页会提示安全问题;webView使用上的建议设置布局高度和宽度设置为 match_parent;具体可见48.开启Google安全浏览服务
07.关于一点面向对象的思想
- 针对webView视频播放演变
- 1.最刚开始把视频全屏show和hide的逻辑都放到X5WebChromeClient中处理,相当于这个类中逻辑比较多
- 2.后期把视频全屏播放逻辑都抽到了VideoWebChromeClient类中处理,这样只需要继承该类即可。这个类独立,拿来即用。
- 3.后期演变,一个视频全屏播放接口 + 接口实现类 + VideoChromeClient,接口主要能够解耦
- 关于webView拦截缓存处理
- 1.代码结构大概是:拦截缓存接口 + 接口实现类 + 接口委派类
- 2.优点:委派类和实现类解耦;便于增加过滤功能(比如用了https+dns优化就不用拦截缓存);
//1.创建委托对象
WebViewCacheDelegate webViewCacheDelegate = WebViewCacheDelegate.getInstance();
//2.通过委托对象调用方法
WebResourceResponse webResourceResponse = webViewCacheDelegate.interceptRequest(url);
- 关于shouldOverrideUrlLoading处理多类型
- 比如:封装库中需要处理打电话,发短信,发邮件,地图定位,图片,超链接等拦截逻辑
- 最刚开始是把处理的逻辑都放到了WebViewClient中的shouldOverrideUrlLoading方法中处理。不过发现这个类代码越来越多……
- 后期演变,针对电话短信等将处理逻辑抽取到WebSchemeIntent类中,针对图片处理逻辑抽取到SaveImageProcessor类中。具体看WebSchemeIntent
- 这样做,相当于保证了类的单一性职责,即类尽量保证内部处理的功能尽可能单一,而不是错综复杂……
08.关于后期需要研究的目标
- 目标
- web页面特别消耗流量,每次打开页面都会请求网络,建议对流量的消耗进行优化……除了对lib库中对拦截做OkHttp缓存,还有什么其他方案
- web页面涉及流量的几个方面
- 普通https请求,一般过程是服务端(对象)-->网络中(二进制流)-->客户端(对象),文本内容会做传输压缩
- 网络图片下载,图片下载消耗的流量较多
- h5页面展示,由于h5页面是交由前端处理显示,客户端开发关注的少些,而此处消耗了大量的流量
- 如何查看web页面消耗流量
- 使用TrafficStats即可查看流量的消耗
WebView库功能完善的更多相关文章
- Fastjson是一个Java语言编写的高性能功能完善的JSON库。
简介 Fastjson是一个Java语言编写的高性能功能完善的JSON库. 高性能 fastjson采用独创的算法,将parse的速度提升到极致,超过所有json库,包括曾经号称最快的jackson. ...
- React-Native 之 GD (十一)加载更多功能完善 及 跳转详情页
1.加载更多功能完善 GDHome.js /** * 首页 */ import React, { Component } from 'react'; import { StyleSheet, Text ...
- android采用MVP完整漫画APP、钉钉地图效果、功能完善的音乐播放器、仿QQ动态登录效果、触手app主页等源码
Android精选源码 一个可以上拉下滑的Ui效果,觉得好看可以学学 APP登陆页面适配 一款采用MVP的的完整漫画APP源码 android实现钉钉地图效果源码 一个使用单个文字生成壁纸图片的app ...
- 我用STM32MP1做了个疫情监控平台4—功能完善界面重新设计
目录 前言 界面展示 新增功能 API 接口说明 多个接口数据的获取和解析 FontAwesome字体图标库的使用 代码下载 系列教程 @ 前言 之前我用STM32MP1和Qt实现了疫情监控平台,系列 ...
- iOS开发UI篇—无限轮播(功能完善)
iOS开发UI篇—无限轮播(功能完善) 一.自动滚动 添加并设置一个定时器,每个2.0秒,就跳转到下一条. 获取当前正在展示的位置. [self addNSTimer]; } -(void)addNS ...
- 微信小程序教学第四章第三节(含视频):小程序中级实战教程:详情-功能完善
详情 - 功能完善 本文配套视频地址: https://v.qq.com/x/page/f0555nfdi14.html 开始前请把 ch4-3 分支中的 code/ 目录导入微信开发工具 这一节中, ...
- C#上位机开发(四)—— SerialAssistant功能完善
上一篇中我们完成了一个串口助手的雏形,实现了基本发送和接收字符串功能,并将打开/关闭串口进行了异常处理,这篇就来按照流程,逐步将功能完善: 1.构思功能 首先是接收部分,要添加一个“清空接收”的按钮来 ...
- 体验ArcGIS9.2的历史库功能
转自原文 体验ArcGIS9.2的历史库功能 ESRI公司于2006年11月9日全球同步发布了历史上重要的软件版本ArcGIS9.2,在该版本中,主要新增了以下四大功能(ESRI田昌莲): 第一大新功 ...
- 漫步Facebook开源C++库Folly之string类设计(散列、字符串、向量、内存分配、位处理等,小部分是对现有标准库和Boost库功能上的补充,大部分都是基于性能的需求而“重新制造轮子”)
就在近日,Facebook宣布开源了内部使用的C++底层库,总称folly,包括散列.字符串.向量.内存分配.位处理等,以满足大规模高性能的需求. 这里是folly的github地址:https:// ...
- android软件简约记账app开发day10-主页面模块--头信息的展示,和之后功能完善的目标。
android软件简约记账app开发day10-主页面模块--头信息的展示,和之后功能完善的目标. 今天来写主界面头信息的展示,也就是将第一天的写的layout中的item_main_top展示到主界 ...
随机推荐
- 机器学习基础03DAY
特征降维 降维 PCA(Principal component analysis),主成分分析.特点是保存数据集中对方差影响最大的那些特征,PCA极其容易受到数据中特征范围影响,所以在运用PCA前一定 ...
- NC200324 魔改森林
题目链接 题目 题目描述 曾经有一道叫做迷雾森林的题目,然而牛牛认为地图中的障碍太多,实在是太难了,所以删去了很多点,出了这道题. 牛牛给出了一个n行m列的网格图 初始牛牛处在最左下角的格点上(n+1 ...
- 从零开始手写 redis(三)内存数据重启后如何不丢失?
前言 我们在 从零手写 cache 框架(一)实现固定大小的缓存 中已经初步实现了我们的 cache. 我们在 从零手写 cache 框架(一)实现过期特性 中实现了 key 的过期特性. 本节,让我 ...
- 在PWM控制下的直流有刷电机性能优化
结论 为了避免各位浪费时间, 先说结论: 选择合适的电机驱动模式和PWM频率, 能大幅提升直流电机的性能和可控性, 在常见的48:1减速电机上, 使用慢衰减模式和低于100Hz的PWM频率, 能达到最 ...
- 李宏毅2022机器学习HW3 Image Classification
Homework3 数据集下载 在本地环境下进行实验总是令人安心,但是又苦于网上找不到数据集,虽然kaggle上有数据集但是下载存在问题 于是有了一个天才的想法,间接从kaggle上下载(利用outp ...
- 突破Windows的极限
偶然碰到这类技术博客,甚感欣慰,但奈何技术水平达不到,很多都难以理解,故记录在此,用作日后学习. 国内有类似的中文翻译,比如:突破Windows极限:物理内存 但是外文链接已经失效,看不到原汁原味的英 ...
- ASP.NET Core MVC应用模型的构建[1]: 应用的蓝图
我个人觉得这是ASP.NET Core MVC框架体系最核心的部分.原因很简单,MVC框架建立在ASP.NET Core路由终结点上,它最终的目的就是将每个Action方法映射为一个或者多个路由终结点 ...
- 使用go module导入本地包
go module是Go1.11版本之后官方推出的版本管理工具,并且从Go1.13版本开始,go module将是Go语言默认的依赖管理工具. 前提 假设我们有learngo和mypackage两个 ...
- 机器学习策略篇:详解单一数字评估指标(Single number evaluation metric)
单一数字评估指标 无论是调整超参数,或者是尝试不同的学习算法,或者在搭建机器学习系统时尝试不同手段,会发现,如果有一个单实数评估指标,进展会快得多,它可以快速告诉,新尝试的手段比之前的手段好还是差.所 ...
- 【LeetCode二叉树#07】左叶子节点之和(基于栈的迭代法前中后序遍历复习)
左叶子节点之和 力扣题目链接(opens new window) 计算给定二叉树的所有左叶子之和. 示例: 思路 注意审题,这里是要求 左叶子节点 之和 不是二叉树中的左侧节点之和,因此使用层序遍历是 ...