webview的一些使用小窍门和需注意的地方

Hybrid App(混合模式移动应用)是指介于web-app、native-app这两者之间的app,兼具“Native App良好用户交互体验的优势”和“Web App跨平台开发的优势”。现如今Hybrid App越来越普遍,越来越受欢迎,主要有以下几点原因:

  • 一方面图文排版上web有着先天的优势,同时纯展示类的页面在目前的移动设备上,性能体验已经很难让用户分辨是网页还是原生了;
  • 另一方面,H5的页面跨平台,方便在原生客户端上实现分享功能,拥有较强的传播性,我们平时常见的活动页面也拥有这样的优势,所以你看到的活动页面也基本都是H5,只需轻轻一点就能分享到各个平台;
  • 同时,H5的页面开发降低了开发成本,一套代码,web,android,ios都能访问。

作为Android开发,就不得不说Webview,WebView 是一个用来显示 Web 网页的控件,继承自 AbsoluteLayout,和使用系统其他控件没什么区别,只是 WeView 控件方法比较多比较丰富,因为它就是一个微型浏览器。我个人根据Webview的内核将Webview分为了三个阶段:

  • 基础阶段:Android 4.3版本之前(包括4.3)Webview采用的是webkit内核,内存小,对于一些图片、视频资源较多的网址就很容易加载不出来,或者直接白屏,但是那个时期也不需要加载太复杂的网页,可以勉强使用
  • 测试阶段:Android 4.4版本,为什么说是测试阶段,因为Android版本4.4是第一次开始使用chrome内核,同时兼容webkit内核,但是兼容的不太好,一堆的问题(具体哪些问题,下面慢慢介绍),所以算是测试阶段,把所有的开发人员当作测试人员
  • 稳定阶段:Android 5.0版本以上(包括5.0)Webview,经过Android 4.4版本的测试,在5.0基本上就已经稳定了,没有什么太大的问题

webview的常用方法

Webview里还是有很多方法的,这里只说一些常用的方法:

//加载网址相关
// 加载URL指定的网页
public void loadUrl(String url);
// 携带http headers加载URL指定的网页
public void loadUrl(String url, Map<String, String> additionalHttpHeaders);
// 使用POST请求加载指定的网页
public void postUrl(String url, byte[] postData);
// 重新加载当前网页
public void reload();
// 加载内容
public void loadData(String data, String mimeType, String encoding);
// 使用baseUrl加载内容
public void loadDataWithBaseURL(String baseUrl, String data, String mimeType, String encoding, String historyUrl);


//网页导航
// 是否可后退
public boolean canGoBack();
// 是否可前进
public boolean canGoForward();
// 是否可前进/后退steps页,大于0表示前进小于0表示后退
public boolean canGoBackOrForward(int steps);
// 后退一页
public void goBack();
// 前进一页
public void goForward();
// 前进/后退steps页,大于0表示前进小于0表示后退
public void goBackOrForward(int steps);


// 其他常用的方法
// 获取当前页面的URL
public String getUrl();
// 获取当前页面的标题
public String getTitle();
// 获取当前页面的加载进度
public int getProgress();
// 清除当前webview访问的历史记录
public void clearHistory();
// 清除网页缓存,由于内核缓存是全局的因此这个方法不仅仅针对webview而是针对整个应用程序
public void clearCache(boolean includeDiskFiles);
// 清除自动完成填充的表单数据
public void clearFormData();
// 类似 Activity 生命周期,页面进入后台不可见状态
public void onPause():
// 在调用 onPause()后,可以调用该方法来恢复 WebView 的运行。
public void onResume():


WebSetting的一些常用方法

WebSettings settings = web.getSettings();

// 存储(storage)
// 启用HTML5 DOM storage API,默认值 false
settings.setDomStorageEnabled(true); 
// 启用Web SQL Database API,这个设置会影响同一进程内的所有WebView,默认值 false
// 此API已不推荐使用,参考:https://www.w3.org/TR/webdatabase/
settings.setDatabaseEnabled(true);  
// 启用Application Caches API,必需设置有效的缓存路径才能生效,默认值 false
// 此API已废弃,参考:https://developer.mozilla.org/zh-CN/docs/Web/HTML/Using_the_application_cache
settings.setAppCacheEnabled(true); 
settings.setAppCachePath(context.getCacheDir().getAbsolutePath());

// 定位(location)
settings.setGeolocationEnabled(true);

// 是否保存表单数据
settings.setSaveFormData(true);
// 是否当webview调用requestFocus时为页面的某个元素设置焦点,默认值 true
settings.setNeedInitialFocus(true);  

// 是否支持viewport属性,默认值 false
// 页面通过`<meta name="viewport" ... />`自适应手机屏幕
settings.setUseWideViewPort(true);
// 是否使用overview mode加载页面,默认值 false
// 当页面宽度大于WebView宽度时,缩小使页面宽度等于WebView宽度
settings.setLoadWithOverviewMode(true);
// 布局算法
settings.setLayoutAlgorithm(WebSettings.LayoutAlgorithm.NORMAL);

// 是否支持Javascript,默认值false
settings.setJavaScriptEnabled(true); 
// 是否支持多窗口,默认值false
settings.setSupportMultipleWindows(false);
// 是否可用Javascript(window.open)打开窗口,默认值 false
settings.setJavaScriptCanOpenWindowsAutomatically(false);

// 资源访问
settings.setAllowContentAccess(true); // 是否可访问Content Provider的资源,默认值 true
settings.setAllowFileAccess(true);    // 是否可访问本地文件,默认值 true
// 是否允许通过file url加载的Javascript读取本地文件,默认值 false
settings.setAllowFileAccessFromFileURLs(false);  
// 是否允许通过file url加载的Javascript读取全部资源(包括文件,http,https),默认值 false
settings.setAllowUniversalAccessFromFileURLs(false);

// 资源加载
settings.setLoadsImagesAutomatically(true); // 是否自动加载图片
settings.setBlockNetworkImage(false);       // 禁止加载网络图片
settings.setBlockNetworkLoads(false);       // 禁止加载所有网络资源

// 缩放(zoom)
settings.setSupportZoom(true);          // 是否支持缩放
settings.setBuiltInZoomControls(false); // 是否使用内置缩放机制
settings.setDisplayZoomControls(true);  // 是否显示内置缩放控件

// 默认文本编码,默认值 "UTF-8"
settings.setDefaultTextEncodingName("UTF-8");
settings.setDefaultFontSize(16);        // 默认文字尺寸,默认值16,取值范围1-72
settings.setDefaultFixedFontSize(16);   // 默认等宽字体尺寸,默认值16
settings.setMinimumFontSize(8);         // 最小文字尺寸,默认值 8
settings.setMinimumLogicalFontSize(8);  // 最小文字逻辑尺寸,默认值 8
settings.setTextZoom(100);              // 文字缩放百分比,默认值 100

// 字体
settings.setStandardFontFamily("sans-serif");   // 标准字体,默认值 "sans-serif"
settings.setSerifFontFamily("serif");           // 衬线字体,默认值 "serif"
settings.setSansSerifFontFamily("sans-serif");  // 无衬线字体,默认值 "sans-serif"
settings.setFixedFontFamily("monospace");       // 等宽字体,默认值 "monospace"
settings.setCursiveFontFamily("cursive");       // 手写体(草书),默认值 "cursive"
settings.setFantasyFontFamily("fantasy");       // 幻想体,默认值 "fantasy"


if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT) {
    // 用户是否需要通过手势播放媒体(不会自动播放),默认值 true
    settings.setMediaPlaybackRequiresUserGesture(true);
}
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.LOLLIPOP) {
    // 5.0以上允许加载http和https混合的页面(5.0以下默认允许,5.0+默认禁止)
    settings.setMixedContentMode(WebSettings.MIXED_CONTENT_ALWAYS_ALLOW);
}
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) {
    // 是否在离开屏幕时光栅化(会增加内存消耗),默认值 false
    settings.setOffscreenPreRaster(false);
}

if (isNetworkConnected(context)) {
    // 根据cache-control决定是否从网络上取数据
    settings.setCacheMode(WebSettings.LOAD_DEFAULT);
} else {
    // 没网,离线加载,优先加载缓存(即使已经过期)
    settings.setCacheMode(WebSettings.LOAD_CACHE_ELSE_NETWORK);
}

WebViewClient回调方法

// 拦截页面加载,返回true表示宿主app拦截并处理了该url,否则返回false由当前WebView处理
// 此方法在API24被废弃,不处理POST请求
public boolean shouldOverrideUrlLoading(WebView view, String url) {
    return false;
}

// 拦截页面加载,返回true表示宿主app拦截并处理了该url,否则返回false由当前WebView处理
// 此方法添加于API24,不处理POST请求,可拦截处理子frame的非http请求
@TargetApi(Build.VERSION_CODES.N)
public boolean shouldOverrideUrlLoading(WebView view, WebResourceRequest request) {
    return shouldOverrideUrlLoading(view, request.getUrl().toString());
}

// 此方法废弃于API21,调用于非UI线程
// 拦截资源请求并返回响应数据,返回null时WebView将继续加载资源
// 注意:API21以下的AJAX请求会走onLoadResource,无法通过此方法拦截
public WebResourceResponse shouldInterceptRequest(WebView view, String url) {
    return null;
}

// 此方法添加于API21,调用于非UI线程
// 拦截资源请求并返回数据,返回null时WebView将继续加载资源
@TargetApi(Build.VERSION_CODES.LOLLIPOP)
public WebResourceResponse shouldInterceptRequest(WebView view, WebResourceRequest request) {
    return shouldInterceptRequest(view, request.getUrl().toString());
}

// 页面(url)开始加载
public void onPageStarted(WebView view, String url, Bitmap favicon) {
}

// 页面(url)完成加载
public void onPageFinished(WebView view, String url) {
}

// 将要加载资源(url)
public void onLoadResource(WebView view, String url) {
}

// 这个回调添加于API23,仅用于主框架的导航
// 通知应用导航到之前页面时,其遗留的WebView内容将不再被绘制。
// 这个回调可以用来决定哪些WebView可见内容能被安全地回收,以确保不显示陈旧的内容
// 它最早被调用,以此保证WebView.onDraw不会绘制任何之前页面的内容,随后绘制背景色或需要加载的新内容。
// 当HTTP响应body已经开始加载并体现在DOM上将在随后的绘制中可见时,这个方法会被调用。
// 这个回调发生在文档加载的早期,因此它的资源(css,和图像)可能不可用。
// 如果需要更细粒度的视图更新,查看 postVisualStateCallback(long, WebView.VisualStateCallback).
// 请注意这上边的所有条件也支持 postVisualStateCallback(long ,WebView.VisualStateCallback)
public void onPageCommitVisible(WebView view, String url) {
}

// 此方法废弃于API23
// 主框架加载资源时出错
public void onReceivedError(WebView view, int errorCode, String description, String failingUrl) {
}

// 此方法添加于API23
// 加载资源时出错,通常意味着连接不到服务器
// 由于所有资源加载错误都会调用此方法,所以此方法应尽量逻辑简单
@TargetApi(Build.VERSION_CODES.M)
public void onReceivedError(WebView view, WebResourceRequest request, WebResourceError error) {
    if (request.isForMainFrame()) {
        onReceivedError(view, error.getErrorCode(), error.getDescription().toString(), request.getUrl().toString());
    }
}

// 此方法添加于API23
// 在加载资源(iframe,image,js,css,ajax...)时收到了 HTTP 错误(状态码>=400)
public void onReceivedHttpError(WebView view, WebResourceRequest request, WebResourceResponse errorResponse) {
}


// 是否重新提交表单,默认不重发
public void onFormResubmission(WebView view, Message dontResend, Message resend) {
    dontResend.sendToTarget();
}

// 通知应用可以将当前的url存储在数据库中,意味着当前的访问url已经生效并被记录在内核当中。
// 此方法在网页加载过程中只会被调用一次,网页前进后退并不会回调这个函数。
public void doUpdateVisitedHistory(WebView view, String url, boolean isReload) {
}

// 加载资源时发生了一个SSL错误,应用必需响应(继续请求或取消请求)
// 处理决策可能被缓存用于后续的请求,默认行为是取消请求
public void onReceivedSslError(WebView view, SslErrorHandler handler, SslError error) {
    handler.cancel();
}

// 此方法添加于API21,在UI线程被调用
// 处理SSL客户端证书请求,必要的话可显示一个UI来提供KEY。
// 有三种响应方式:proceed()/cancel()/ignore(),默认行为是取消请求
// 如果调用proceed()或cancel(),Webview 将在内存中保存响应结果且对相同的"host:port"不会再次调用 onReceivedClientCertRequest
// 多数情况下,可通过KeyChain.choosePrivateKeyAlias启动一个Activity供用户选择合适的私钥
@TargetApi(Build.VERSION_CODES.LOLLIPOP)
public void onReceivedClientCertRequest(WebView view, ClientCertRequest request) {
    request.cancel();
}

// 处理HTTP认证请求,默认行为是取消请求
public void onReceivedHttpAuthRequest(WebView view, HttpAuthHandler handler, String host, String realm) {
    handler.cancel();
}

// 通知应用有个已授权账号自动登陆了
public void onReceivedLoginRequest(WebView view, String realm, String account, String args) {
}
// 给应用一个机会处理按键事件
// 如果返回true,WebView不处理该事件,否则WebView会一直处理,默认返回false
public boolean shouldOverrideKeyEvent(WebView view, KeyEvent event) {
    return false;
}

// 处理未被WebView消费的按键事件
// WebView总是消费按键事件,除非是系统按键或shouldOverrideKeyEvent返回true
// 此方法在按键事件分派时被异步调用
public void onUnhandledKeyEvent(WebView view, KeyEvent event) {
    super.onUnhandledKeyEvent(view, event);
}

// 通知应用页面缩放系数变化
public void onScaleChanged(WebView view, float oldScale, float newScale) {
}

WebChromeClient回调方法

// 获得所有访问历史项目的列表,用于链接着色。
public void getVisitedHistory(ValueCallback<String[]> callback) {
}

// <video /> 控件在未播放时,会展示为一张海报图,HTML中可通过它的'poster'属性来指定。
// 如果未指定'poster'属性,则通过此方法提供一个默认的海报图。
public Bitmap getDefaultVideoPoster() {
    return null;
}

// 当全屏的视频正在缓冲时,此方法返回一个占位视图(比如旋转的菊花)。
public View getVideoLoadingProgressView() {
    return null;
}

// 接收当前页面的加载进度
public void onProgressChanged(WebView view, int newProgress) {
}

// 接收文档标题
public void onReceivedTitle(WebView view, String title) {
}

// 接收JavaScript控制台消息
public boolean onConsoleMessage(ConsoleMessage consoleMessage) {
    return false;
}

// 接收图标(favicon)
public void onReceivedIcon(WebView view, Bitmap icon) {
}

// Android中处理Touch Icon的方案
// http://droidyue.com/blog/2015/01/18/deal-with-touch-icon-in-android/index.html
public void onReceivedTouchIconUrl(WebView view, String url, boolean precomposed) {
}

// 通知应用当前页进入了全屏模式,此时应用必须显示一个包含网页内容的自定义View
public void onShowCustomView(View view, CustomViewCallback callback) {
}

// 通知应用当前页退出了全屏模式,此时应用必须隐藏之前显示的自定义View
public void onHideCustomView() {
}


// 显示一个alert对话框
public boolean onJsAlert(WebView view, String url, String message, JsResult result) {
    return false;
}

// 显示一个confirm对话框
public boolean onJsConfirm(WebView view, String url, String message, JsResult result) {
    return false;
}

// 显示一个prompt对话框
public boolean onJsPrompt(WebView view, String url, String message, String defaultValue, JsPromptResult result) {
    return false;
}

// 显示一个对话框让用户选择是否离开当前页面
public boolean onJsBeforeUnload(WebView view, String url, String message, JsResult result) {
    return false;
}


// 指定源的网页内容在没有设置权限状态下尝试使用地理位置API。
// 从API24开始,此方法只为安全的源(https)调用,非安全的源会被自动拒绝
public void onGeolocationPermissionsShowPrompt(String origin, GeolocationPermissions.Callback callback) {
}

// 当前一个调用 onGeolocationPermissionsShowPrompt() 取消时,隐藏相关的UI。
public void onGeolocationPermissionsHidePrompt() {
}

// 通知应用打开新窗口
public boolean onCreateWindow(WebView view, boolean isDialog, boolean isUserGesture, Message resultMsg) {
    return false;
}

// 通知应用关闭窗口
public void onCloseWindow(WebView window) {
}

// 请求获取取焦点
public void onRequestFocus(WebView view) {
}

// 通知应用网页内容申请访问指定资源的权限(该权限未被授权或拒绝)
@TargetApi(Build.VERSION_CODES.LOLLIPOP)
public void onPermissionRequest(PermissionRequest request) {
    request.deny();
}

// 通知应用权限的申请被取消,隐藏相关的UI。
@TargetApi(Build.VERSION_CODES.LOLLIPOP)
public void onPermissionRequestCanceled(PermissionRequest request) {
}

// 为'<input type="file" />'显示文件选择器,返回false使用默认处理
@TargetApi(Build.VERSION_CODES.LOLLIPOP)
public boolean onShowFileChooser(WebView webView, ValueCallback<Uri[]> filePathCallback, FileChooserParams fileChooserParams) {
    return false;
}

别看我上面列举出来了那么多的方法,其实当我们真正使用webview的时候,根本就用不到那么多,就拿webview的方法来说,我们通常也就只会用到load()和goback()方法,而webSetting一般都是设置的默认值,需要js交互的话,可能会需要另外设置一些属性,两个监听回调也是一样的,只用个别的方法。

有关WebView的一些问题

1、什么时候WebView算是加载完成

相信所有使用过webview的Android开发人员都会有这个困惑,虽然在WebViewClient回调里面有onPageFinished()方法,可是在我们自己使用的时候就会发现,其实当onPageFinished()方法回调的时候,webview并没有完全的加载完成,而有些操作,特别是与js交互的时候又需要在网页加载完成的情况下在进行,应该怎么办呢?

来看一下当一个webview加载网页时的回调方法:

09-17 12:52:29.711 24061-24061/com.zl.webview.demo.myapplication D/MainActivity: onProgressChanged(0)
09-17 12:52:29.741 24061-24061/com.zl.webview.demo.myapplication D/MainActivity: onPageStarted
09-17 12:52:29.966 24061-24061/com.zl.webview.demo.myapplication D/MainActivity: onProgressChanged(22)
09-17 12:52:30.081 24061-24061/com.zl.webview.demo.myapplication D/MainActivity: onProgressChanged(53)
09-17 12:52:30.126 24061-24061/com.zl.webview.demo.myapplication D/MainActivity: onReceivedTitle
09-17 12:52:30.611 24061-24061/com.zl.webview.demo.myapplication D/MainActivity: onProgressChanged(100)
09-17 12:52:30.636 24061-24061/com.zl.webview.demo.myapplication D/MainActivity: onPageFinished

通过打印出来的方法顺序可以看到,确实是onPageFinished()方法是最后在回调的,那么是不是就应该把网页加载完成的处理就放在onPageFinished()方法里面呢?来再让我们看一下Android 4.4版本的手机会怎样:

09-17 12:53:17.096 24061-24061/com.zl.webview.demo.myapplication D/MainActivity: onProgressChanged(0)
09-17 12:53:17.111 24061-24061/com.zl.webview.demo.myapplication D/MainActivity: onPageStarted
09-17 12:53:17.181 24061-24061/com.zl.webview.demo.myapplication D/MainActivity: onProgressChanged(22)
09-17 12:53:17.246 24061-24061/com.zl.webview.demo.myapplication D/MainActivity: onReceivedTitle
09-17 12:53:17.381 24061-24061/com.zl.webview.demo.myapplication D/MainActivity: onProgressChanged(72)
09-17 12:53:17.436 24061-24061/com.zl.webview.demo.myapplication D/MainActivity: onProgressChanged(100)
09-17 12:53:17.466 24061-24061/com.zl.webview.demo.myapplication D/MainActivity: onPageFinished
09-17 12:53:17.512 24061-24061/com.zl.webview.demo.myapplication D/MainActivity: onPageFinished

可以看到在Android 4.4版本的手机上,onPageFinished()方法是回调了两次的,所以如果你有什么两次调用并不影响的操作,可以放到onPageFinished()方法里面,如果两次调用会影响的话,就放到onProgrossChanged()方法里面,当进度之大于等于100的时候在调用方法就好了。

2、WebView的标题应该怎样获取

可能大多数人一下子就想到了onReciviedTitle()方法,因为这个回调方法直接就返回了网页的title,完全可以直接用。当然这样想并没有错,但是当王爷内部有跳转需要回退怎么办,来我们再看一下,当webview调用了goBack()方法后,回调会怎么走:

09-17 12:55:35.983 14678-14678/com.zl.webview.demo.myapplication D/MainActivity: onProgressChanged(0)
09-17 12:55:36.003 14678-14678/com.zl.webview.demo.myapplication D/MainActivity: onPageStarted
    onProgressChanged(42)
09-17 12:55:36.033 14678-14678/com.zl.webview.demo.myapplication D/MainActivity: onProgressChanged(100)
09-17 12:55:36.053 14678-14678/com.zl.webview.demo.myapplication D/MainActivity: onPageFinished

可以看到,当webview回退到上一个网页的时候,并不会再回调onReciviedTitel()方法,所以标题在这种情况下并不会在发生变化了,其实将标题的改变放到onPageFinished()方法里面就好了,直接调用webview自带的getTitle()方法就行了。

3、为什么设置了WebSetting里面设置了缩放,内容还是没有全屏

遇到这种情况,首先确定自己是否将websetting里面有关缩放的设置都有过设置,有的话,就拿到网址信息照相关的前端开发人员,让他们去掉网页里面div标签的定死的宽度和高度值,改为自适应即可。

4、一个网页的图片加载不出来

这个问题是我之前在某个项目开发过程中遇到的,当时很费解,webview的各项配置什么的,都仔仔细细的检查了一遍又一遍,断点调试了好多遍也没有找到原因。然后偶然的看到了WebSetting的一个设置,

settings.setBlockNetworkImage(true);

猜想问题可能就出在这里,这个方法是设置是否加载网络图片,然后按照这个思路去试了一下,果然图片可以正常加载了。

5、页面关闭后音频、视频还在后台播放

是因为home键后,并没有暂停掉webview里面的media播放,需要在Activity里面作如下处理:


    @Override
    public void onResume() {
        super.onResume();
        if (webView != null && Build.VERSION.SDK_INT >= Build.VERSION_CODES.HONEYCOMB) {
            webView.onResume();
        }
    }

    @Override
    public void onPause() {
        super.onPause();
        if (webView != null && Build.VERSION.SDK_INT >= Build.VERSION_CODES.HONEYCOMB) {
            webView.onPause();
        }
    }

当然,大家在查看源码的时候,还发现了pauseTimers()和resumeTimers()两个方法,和图片中的方法很像,这两个方法的作用和那两个方法是一样的不过作用域不一样,pauseTimers()和resumeTimers()是针对于应用内部的所有的webview的,而onPause()和onResume()只针对于当前的webview。

6、WebView滑动到底部的判断

Android判断WebView是否已经滚动到页面底端:
getScrollY()方法返回的是当前可见区域的顶端距整个页面顶端的距离,也就是当前内容滚动的距离. 
getHeight()或者getBottom()方法都返回当前WebView 这个容器的高度 
getContentHeight 返回的是整个html 的高度,但并不等同于当前整个页面的高度,因为WebView 有缩放功能, 所以当前整个页面的高度实际上应该是原始html 的高度再乘上缩放比例. 因此,更正后的结果,准确的判断方法应该是: 
if(WebView.getContentHeight*WebView.getScale() == (webview.getHeight()+WebView.getScrollY())){ //已经处于底端 } 

7、网页本身的错误怎么看

在webview和js交互的时候,有时候经常会出现,调js方法调不通,或者js调Android方法不同的情况,打个断点调试也很难发现问题,也完全不清楚问题出在那边,仔细看自己的代码又觉得没有问题,但是网页上现在自己也没法调试,找前端的网页开发也补一点给能够找到问题出在哪里,这个时候就可以用WebClientChorme里面的一个回调方法onConsoleMessage(ConsoleMessage consoleMessage),其中ConsoleMessage兑现就和logcat打印的语句一样,也分ERROR,DEBUG,INFO等等级,我们可以根据自己的情况来打印自己想要的信息,老看网页本身有没有错误。

8、关于网页里面广告过滤的问题

首先我们得先弄面白为什么会在网页中出现广告,如果是前端网页设计本身就有的招租的广告位,那就没有什么大惊小怪的,但如果是自己冒出来的那就要注意了,你的网址被劫持了,一般网址被劫持费为三种情况:

  1. dns劫持,

    DNS是在作为域名(www.baidu.com)和IP地址(192.168.1.1)相互映射的一个分布式数据库,就是我们的浏览器,会将域名拿到DNS去解析出ip地址来访问,DNS劫持是指在劫持的网络范围内拦截域名解析的请求,分析请求的域名,把审查范围以外的请求放行,否则返回假的IP地址或者什么都不做使请求失去响应,其效果就是对特定的网络不能反应或访问的是假网址。

    通俗讲,DNS帮我们指向了另一个地址,或者让我无法访问

  2. http劫持,

    HTTP劫持是在使用者与其目的网络服务所建立的专用数据通道中,监视特定数据信息,提示当满足设定的条件时,就会在正常的数据流中插入精心设计的网络数据报文,目的是让用户端程序解释“错误”的数据,并以弹出新窗口的形式在使用者界面展示宣传性广告或者直接显示某网站的内容。

    通俗讲,你要去别人家的首页,他会给你别人家的首页,但却在别人家首页某个部位加个小广告,还可以控制关掉。。 让人无奈。现在很多免费WIFI植入广告的手段,为了博取更多点击量。

  3. 第三方jar包植入广告,这种情况就没法弄了

通常情况下,网址被劫持了,将http换成https就可以了,但是有时候公司需求什么的乱七八糟的原因,不愿意使用https的网址,这是就只能够在网页加载的时候想办法过滤广告了,我这里有两种思路参考:

1、对加载的域名进行判断,不是公司常用的域名,就不做加载的操作

2、对加载的网址进行判断,如果是常见的那些个广告的域名,就不加载(在百度上可以找到常见的广告域名)

不管用哪一种思路都需要使用到WebViewClient毁掉里面的shouldInterceptRequest()方法,不同的是,两种思路的判断不同

@Override
    public WebResourceResponse shouldInterceptRequest(WebView view, String url) {
        url = url.toLowerCase();
        if () {//在这里判断URL里面使用的是公司常见的域名或者是常见的广告的域名来京行不通的处理
            return super.shouldInterceptRequest(view, url);//正常加载
        }else{
            return new WebResourceResponse(null,null,null);//含有广告资源屏蔽请求
        }
    }

9、关于WebView内存泄露的问题

关于这个问题,我很难给你一个清晰的描述,你在谷歌里搜 webview lead memory 能搜到很多结果 甚至还有给谷歌提交的issue 哈哈,我也无法给出一个清晰的答案 在什么时候 什么版本那些手机上一定会出现内存泄露,但是根据我自己的monkey结果来看,有时,webview内存泄露的情况还是很严重的,尤其是当你加载的页面比较庞大的时候。解决方案 我查了很多也用了很多,但是都不太理想,最后看了下微信和qq的做法,试了一下是目前效果最好的,就是在Activity销毁之前,尽可能的去释放掉webview所持有的一些东西,并将webview置空。

10、WebView的缓存机制

1. WebView缓存分为:页面缓存和数据缓存。页面缓存指加载网页时,对页面或资源数据的缓存。一般使用RE管理器进入目录: “/data/data/(packageName)/cache/org.chromium.android_webview“可看到;
数据缓存又分为 AppCache 与 DOM Storage 。AppCache可以有选择地缓存我们所想要缓存的东西;DOM Storage 则是HTML5的一个缓存机制,常用于存储简单的表单数据。
2. webView的缓存模式:
LOAD_CACHE_ONLY: 不使用网络,只读取本地缓存数据
LOAD_DEFAULT: 根据cache-control决定是否从网络上取数据。
LOAD_CACHE_NORMAL: API level 17中已经废弃, 从API level 11开始作用同LOAD_DEFAULT模式
LOAD_NO_CACHE: 不使用缓存,只从网络获取数据.
LOAD_CACHE_ELSE_NETWORK,只要本地有,无论是否过期,或者no-cache,都使用缓存中的数据。
3. 将缓存路径转移到外置sd卡

 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值