Android WebView重定向的网页无法goBack()的几个解决方案

如有转载,请声明出处: 时之沙: http://blog.csdn.net/t12x3456

Android WebView 因重定向无法正常goBack()解决方案

 首先说下问题,初始页面为A,点击某个链接跳转到B(http://xxx.com.cn/),B页面重定向到C页面(http://xxx.com.cn/website/index.html)

当调用webview.goBack()时,页面回退到B,然后接着会重定向回C页面.

这样会导致两个问题:

1. 无法回退到webview的初始页面A

2. 无法正常退出Activity或者Fragment(只有还未加载完C时进行回退才能退出页面)

关于如何解决这个问题,我总结了如下三种方法,可以根据具体情况进行使用:

一. 首先需要和前端开发人员沟通,看重定向是否必要,如果跳转链接只是域名,然后默认重定向到  域名/index.html,并没有特殊处理的话,那么这种重定向并没有意义.

只要将网页中的连接,比如

[html] view plain copy 在CODE上查看代码片 派生到我的代码片
  1. <a href="http://xxx.com.cn"/>  
直接替换为
[html] view plain copy 在CODE上查看代码片 派生到我的代码片
  1. <a href="http:///xxx.com.cn/index.html"/>  
即可解决该问题.

二.页面中的重定向是必须的,那么我们就需要自己维护一个webview的历史栈,根据自己的需求进行过滤跳转或者重新加载页面:

判断到当前为重定向后的链接,那么那么当回退的时候就需要忽略上一级的链接,不使用webview.goback(),移除重定向和重定向后的url,

获取到初始页面链接后自己进行loadUrl()操作.

3.还有一种方法,和方法2类似,需要自己维护webview的历史栈,但是需要前端的配合,提供js函数获取网页是否进行重定向

在webviewClient回调shouldoverloading()中过滤url时,若属于重定向的地址,则不加入栈中,回退时根据历史栈加载即可.

这里主要讲一下方法二:

首先定义一个历史栈 :

[html] view plain copy 在CODE上查看代码片 派生到我的代码片
  1. private ArrayList<String> loadHistoryUrls = new ArrayList<String>();  
把初始页面Url加入
[html] view plain copy 在CODE上查看代码片 派生到我的代码片
  1. loadHistoryUrls.add(INITAL_WEB_URL);  
然后加入加载的url:
[java] view plain copy 在CODE上查看代码片 派生到我的代码片
  1. public boolean shouldOverrideUrlLoading(WebView view,String url){  
  2.   
  3.         //将过滤到的url加入历史栈中  
  4.           loadHistoryUrls.add(url);  
  5.           return true;            
  6.   
  7.        }  

最后在webview.goback()处理:

[java] view plain copy 在CODE上查看代码片 派生到我的代码片
  1. @Override  
  2.     public boolean onKeyDown(int keyCode, KeyEvent event) {  
  3.         //判断是否可以返回操作  
  4.         if (webView.canGoBack() && event.getKeyCode() == KeyEvent.KEYCODE_BACK) {  
  5.            //过滤是否为重定向后的链接    
  6.            if(loadHistoryUrls.size()>0&&loadUrls.get(loadHistoryUrls.size()-1).contains("index.html"))  
  7.   
  8.              //移除加载栈中的最后两个链接  
  9.              loadHistoryUrls.remove(loadHistoryUrls.get(loadHistoryUrls.size()-1));  
  10.   
  11.              loadHistoryUrls.remove(loadHistoryUrls.get(loadHistoryUrls.size()-1));  
  12.   
  13.              //加载重定向之前的页  
  14.              webview.load(loadUrls.get(loadHistoryUrls.size()-1));  
  15.    
  16.             return true;  
  17.             }   
  18.             
  19.         }          
  20.   }  

关于加载栈,后来发现webview本身也有对应的API:

[java] view plain copy 在CODE上查看代码片 派生到我的代码片
  1. //获取历史列表  
  2.  WebBackForwardList mWebBackForwardList = webView.copyBackForwardList();  
不过这个api可能受系统版本的影响或者不同手机系统进行了修改

所以解决该问题时,大家可以自己根据需求,自己维护加载的历史栈或者直接调用系统api.

这里总结一下,若重定向非必要,采取方案一,最简单,修改量也非常小. 重定向必要,则使用方案二或者方案三.

因为需要和前端人员交互,方案三所需要的沟通,开发,维护的成本要比方案二高出不少,但对于是否重定向的判断非常准确,若有多个重定向的情况,一次开发完成后不需要对代码再次改动.  方案二则需要写死需要过滤的url,若出现多个重定向,则会显得代码比较臃肿,每次都需要重新增加代码. 具体使用依据项目中的开发情况而定.

       最后再补充一种通用的办法,但是需要后台的强大支持: 在webview进行加载时,将请求发送至服务器,然后由服务器进行分析处理,将处理后的结果返回给客户端进行显示. 并且可以由服务器对网页内容进行编码或者取出冗余,并结合cdn提升响应速度,这也是目前浏览器开发常用的一种策略.但是需要大量的数据收集,分析和处理,对于服务器的依赖比较严重,若开发进度较紧或者公司资源有限,可先参照以上办法进行解决.

     最重还要讲的一点, 本篇文章主要是对于加载己方开发的H5中遇到问题的解决,至于第三方网站加载,这个是没有办法解决的. 包括微信上也一样,对于各种公众平台和第三方链接,是没有通用解决方案的, 所以他们在交互上进行了处理 ,在H5进行一次跳转就会在标题栏左上角出现关闭按钮. 毕竟用户是不知道快速连续点击两次返回才能正常返回首页的.

如有转载,请声明出处: 时之沙: http://blog.csdn.net/t12x3456

public boolean onKeyDown(int keyCode, KeyEvent event) { //判断是否可以返回操作 if (webView.canGoBack() && event.getKeyCode() == KeyEvent.KEYCODE_BACK) { //获取历史列表 WebBackForwardList mWebBackForwardList = webView .copyBackForwardList(); //判断当前历史列表是否最顶端,其实canGoBack已经判断过 if (mWebBackForwardList.getCurrentIndex() > 0) { //获取历史列表 String historyUrl = mWebBackForwardList.getItemAtIndex( mWebBackForwardList.getCurrentIndex() - 1).getUrl(); //按照自己规则检查是否为可跳转地址 //注意:这里可以根据自己逻辑循环判断,拿到可以跳转的那一个然后webView.goBackOrForward(steps) if (!historyUrl.contains("s.click.taobao.com")) { //执行跳转逻辑 webView.goBack(); //webView.goBackOrForward(-1) return true; } } } return super.onKeyDown(keyCode, event); }
  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
WebView第二次调用loadUrl时,页面不会刷新的原因可能是由于缓存导致的。因为WebView默认启用了缓存机制,当再次加载同一个URL时,会直接从缓存中读取,而不会重新请求服务器。 要解决这个问题,可以通过以下两种方式: 1.禁用WebView缓存 在调用loadUrl之前,可以通过设置WebSettings对象禁用缓存: ```java webView.getSettings().setCacheMode(WebSettings.LOAD_NO_CACHE); ``` 这样每次加载URL时,都会强制从服务器重新获取数据,并禁用缓存。 2.强制刷新WebViewWebView中添加一个刷新按钮,每次点击刷新按钮时,强制刷新WebView。具体实现如下: ```java // 添加一个刷新按钮 Button refreshBtn = findViewById(R.id.refresh_btn); refreshBtn.setOnClickListener(new View.OnClickListener() { @Override public void onClick(View v) { webView.reload(); } }); // 在WebViewClient中监听页面加载完成事件,并隐藏刷新按钮 webView.setWebViewClient(new WebViewClient() { @Override public void onPageFinished(WebView view, String url) { super.onPageFinished(view, url); refreshBtn.setVisibility(View.GONE); } @Override public void onReceivedError(WebView view, int errorCode, String description, String failingUrl) { super.onReceivedError(view, errorCode, description, failingUrl); refreshBtn.setVisibility(View.VISIBLE); } }); ``` 当页面加载完成时,隐藏刷新按钮;当页面加载失败时,显示刷新按钮。这样就可以通过点击刷新按钮来强制刷新WebView了。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值