Android性能优化(1)---webview优化篇

写这篇文章的主要是因为在实际应用的时候遇到了很多棘手的问题(文章最后部分,交流一下遇到的问题),在这里分享出来让人少走一些弯路,如果有不对的地方直接留言即可

相信一般做Android开发的同学都会或多或少的使用webview,(由于不太了解js,js交互的部分暂时没有)相信有不少人的使用过程是这样的

布局
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:orientation="vertical">

    <WebView
        android:id="@+id/webView"
        android:layout_width="match_parent"
        android:layout_height="match_parent" >
    </WebView>
</LinearLayout>
使用
    //设置webview
    WebSettings websettings = webView.getSettings();
    websettings.setJavaScriptEnabled(true);
    websettings.setDomStorageEnabled(true);
    websettings.setJavaScriptCanOpenWindowsAutomatically(true);
    websettings.setBuiltInZoomControls(false);// 
    webView.loadUrl(url);

上边这样使用,在使用量少的情况下可能没有问题,但是也给程序留下了不少隐患--内存泄漏,由于webview并没有完全清除掉,一直持有对应的上下文,导致内存无法被回收

说道这里就有了第一种的优化

优化一

既然说没有主动回收,那我们就手动回收在,activity的onDestroy方法中清除webview

public void onDestroy() {
    super.onDestroy();
    webView.clearHistory();
    webView.clearView();
    webView.removeAllViews();
    webView.destroy();
}

当你做完这些操作后,退出当前的activity,发现内存好像并没有减少(或者浮动不大)原因请参考
http://blog.csdn.net/u013085697/article/details/53259116

优化二

在需要用到webview的地方直接new一个出来,而不是在.xml文件中定义webview

布局

在布局需要使用webview的地方使用一个FrameLayout代替(其他的容器也可以)

使用
/**
*创建webview并添加到布局容器中
*/
public void onCreate(Bundle savedInstanceState) {
   super.onCreate(savedInstanceState);
   WebView webView = WebviewUtil.createNewWebView(context);
   FrameLayout frameLayout  = findViewById(R.id.xxx);
   frameLayout.addView(mWebView);
}
...
/**
*创建webview公共方法
*/
public static android.webkit.WebView createNewWebView(Context context){
    android.webkit.WebView webView = new android.webkit.WebView(context.getApplicationContext());
    webView.setLayerType(View.LAYER_TYPE_SOFTWARE, new Paint());
    webView.setLayoutParams(new FrameLayout.LayoutParams(FrameLayout.LayoutParams.MATCH_PARENT, FrameLayout.LayoutParams.MATCH_PARENT));
    return webView;
}

/**
*清除webview
*/
public void onDestroy() {
 if (null != webView) {
        try {
            webView.removeJavascriptInterface("xx");
            webView.getSettings().setJavaScriptEnabled(false);
            webView.loadUrl("about:blank");
            webView.loadDataWithBaseURL(null, "", "text/html", "uft-8", null);
            ViewParent parent = webView.getParent();
            if (parent != null) {
                ((ViewGroup) parent).removeView(webView);
            }
            webView.clearHistory();
            webView.clearView();
            webView.removeAllViews();
            webView.destroy();
        } catch (Exception e) {
            e.printStackTrace();
        }
    }
    super.onDestroy();
   
}

但是通过以上操作,查看内存并没有被回收.........

优化三 -- 开进程

当你的webview需要的内存很大,但是主进程申请不到更多的内存的情况下可以使用。

此方法各大主流APP基本都是这样操作的,在打开webview页面后,查看此应用的进程是否多开了一个。

开进程的方法也很简单只需要在你需要的activity在AndroidManifest中注册的时候添加“process”属性即可,但是注意进程间的通信;当关闭此activity时杀掉此进程不然内存不能被回收,下次进入时内存累加,最终会导致溢出

    <activity
        android:name=".Activity"
        android:process=":remoteView"
       />

如果打开进程的速度慢可以先进行进程的预加载,先启动一个和上边activity同一个进程的service,然后再打开activity

优化四--使用第三方

腾讯的x5内核,具体的优势可以参考腾讯官方文档

最后,再来说说我遇到的问题吧,首先需求是这样的一个页面需要多个统计图(百度的echarts,为了保持两端一致性和多样化的图表需求)每一个图表都需要一个对应的webview

没办法,产品提出来了

1开始使用第一种,崩溃,卡顿问题不断

2使用方法二,解决了大部分的机型,只有蓝绿大厂不行(吐槽中OPPO、vivo)
会出现无故的崩溃,查看logo日志(包括sdk底层logo日志)都只能找到一个内存地址的错误,比如:

Fatal signal 11 (SIGSEGV), code 1, fault addr 0x9ffffffff in tid 30218 (Thread-708)

没有其他任何提示,腾讯的bugly,友盟也没有抓取到相关的信息,崩溃中
后来查到两点问题

1、echarts的问题,其中有一个svg渲染(官方是beta版),是为了给低性能设备使用,看情况使用,有些图表会出现点击蓝屏闪烁(只是在webview范围之内),不过亲测内存降低了太多,没有使用前一个带滑块的图表达到了快200M,改为svg之后一直维持在30+M,提升明显,如果使用请自行测试多种机型

2、还是echarts 的问题,数据返回后多次向webview传入了数据导致崩溃,改为一次后暂未发现

以上两个问题目前只在OPPO和vivo手机发现,希望有此需求的小伙伴注意此问题

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值