为什么返回404?
当我们使用WebView加载一个html页面时,通常会在WebViewClient 的onReceivedHttpError()与onReceivedError()去做一些错误响应的处理,但是有时候虽然页面加载成功,onReceivedHttpError()这个方法却会返回404,为什么会返回404呢?WevView是Android系统内置的一个浏览器,同别的浏览器一样,WebView在请求加载一个页面的同时,还会发送一个请求图标文件的请求。
比如我们采用WebView去加载一个页面:
webView.loadUrl("http://192.168.5.40:9006/sso_web/html/H5/doctor/aboutUs.html");
同时还会发送一个请求图标文件的请求
http://192.168.5.40:9006/favicon.ico
onReceivedHttpError这个方法主要用于响应服务器返回的Http错误(状态码大于等于400),这个回调将被调用任何资源(IFRAME,图像等),而不仅仅是主页面。所以就会出现主页面虽然加载成功,但由于网站没有favicon.ico文件导致返回404错误。
如何处理?
1.重写WebViewClient的shouldInterceptRequest方法禁用favicon.ico请求
@RequiresApi(api = Build.VERSION_CODES.LOLLIPOP) @Override public WebResourceResponse shouldInterceptRequest(WebView view,
WebResourceRequest request) {
if(!request.isForMainFrame() && request.getUrl().getPath().endsWith("/favicon.ico")) {
try {
return new WebResourceResponse("image/png", null,
new BufferedInputStream(view.getContext().getAssets().open("empty_favicon.ico")));
} catch (Exception e) {
e.printStackTrace();
}
}
return null;
}
@Override public WebResourceResponse shouldInterceptRequest(WebView view, String url) {
if(url.toLowerCase().contains("/favicon.ico")) {
try {
return new WebResourceResponse("image/png", null,
new BufferedInputStream(view.getContext().getAssets().open("empty_favicon.ico")));
} catch (Exception e) {
e.printStackTrace();
}
}
return null;
}
为了兼容5.0之前的版本,所以需要重写俩个方法,当我们使用WebView加载网页任何资源时,都会回掉
shouldInterceptRequest方法,所以借助shouldInterceptRequest方法可以帮助我们进行资源加载请求拦截,返回我们自己构造的WebResourceResponse,如果返回null,WebView则会继续加载资源,Ps:这个方法在非UI线程中执行,所以尽量避免执行UI相关操作。
2.在onReceivedHttpError()做相应处理
@Override
public void onReceivedHttpError(WebView view, WebResourceRequest request,
WebResourceResponse errorResponse) {
if (!request.isForMainFrame() && request.getUrl().getPath().endsWith("/favicon.ico") ) {
Log.e(TAG,"favicon.ico 请求错误"+errorResponse.getStatusCode()+errorResponse.getReasonPhrase());
} else {
// TODO: 具体可根据返回状态码做相应处理
ToastUtil.showToast(getApplicationContext(),errorResponse.getReasonPhrase());
}
super.onReceivedHttpError(view, request, errorResponse);
}
onReceivedHttpError方法中有俩个重要参数,分别为request与errorResponse,通过request参数可以知道请求类型以及请求的url,通过errorResponse参数可以获得返回的状态码以及错误原因描述,然后我们可以根据相应的请求参数以及返回状态去做相应的错误处理。Ps:上面代码仅仅作为一个简单的示例,具体实现可根据产品需求去实现,比如toast提醒、显示一个错误的页面等等。