LeakCanary提示` policy.HwPhoneWindow$1.this$0`的泄漏分析

最新在华为手机GEM=703L android6.0发现的问题,在AsyncTask执行ProgressDialog的显示或隐藏,然后退出activity会发生泄漏。泄漏提示
GC ROOT com.android.internal.policy.HwPhoneWindow$1.this$0

开始分析

首先上mat的分析图

Paste_Image.png

其中提示了两个未清除的引用,都指向了HwPhoneWindow,貌似其内部类引用了它,这是什么东东?
凭我有限的小脑分析,涉及Window类在我的app只有两种:Activity,Dialog。Activity我已经完美的解决了泄漏问题,那先从Dialog下手。

在Dialog类中可以看到可以看到mWindow,mDecor

  final Window w = new PhoneWindow(mContext);
        mWindow = w;

...
mDecor = mWindow.getDecorView();

想到了什么,是不是跟Activity很像,原来Dialog自己拥有WIndow并维护,但是新建的时候使用Activity的上下文,在Activity销毁的时候,Dialog不销毁就会有泄漏风险,而且Dialog的生命周期会跟Activity产生不同步。

尝试1

原来写法

 progressDialog = new ProgressDialog(Activity.this);
 progressDialog.setProgress(ProgressDialog.STYLE_SPINNER);
 progressDialog.setTitle("加载通讯录中...");

并在Activity的onDestory执行super之前销毁

@Override
    protected void onDestroy() {
        if(progressDialog != null ){
            progressDialog.dismiss();
            progressDialog = null;
        }
        super.onDestroy();
    }

结果:
还是存在泄漏

尝试2

修改原来的写法,使用FragmentDialog改造,利用Fragment维护Dialog的生命周期
改造后:

public class ProgressFragmentDialog extends DialogFragment {
 @Override
    public Dialog onCreateDialog(Bundle savedInstanceState) {
        AlertDialog.Builder progressDialog = new AlertDialog.Builder(this.getActivity());
        progressDialog.setView(getActivity().getLayoutInflater().inflate(R.layout.dialog_progress,null));
        progressDialog.setCancelable(false);


        //取消返回键
        progressDialog.setOnKeyListener(new DialogInterface.OnKeyListener() {
            @Override
            public boolean onKey(DialogInterface dialog, int keyCode, KeyEvent event) {

                return true;
            }
        });
        AlertDialog dialog = progressDialog.create();
        dialog.setCanceledOnTouchOutside(false); //取消点击外关闭

        return dialog;
    }
}

//显示Dialog
 progressFragmentDialog = new ProgressFragmentDialog();
        progressFragmentDialog.show(this.getFragmentManager(),"ProgressFragmentDialog");

结果:

还是存在泄漏

what fuck! 分析到这,我已经黔驴技穷了,干脆不用Dialog,自己实现弹出框

尝试3

思路就是拿到根节点decorView,new一个View adddecorView上,前提是在Activity的setContentView()之后执行
上代码

public class ProgressViewDialog {

    /**
     * 上下文,存储activity信息
     */
   // private Context context;
    private ViewGroup decorView; //decorView
   // private ViewGroup activityRootView;//内容区域的根视图
    private ViewGroup dialogView;//我的根视图


    /**
     * 构造函数
     * @param context
     */
    public ProgressViewDialog(Context context)
    {
        //获得一个xml布局加载器
        LayoutInflater layoutInflater = LayoutInflater.from(context);

        //获得decorView
        decorView = (ViewGroup)((Activity)context).getWindow().getDecorView();
        //Log.d("decorView count", decorView.getChildCount()+"") ;
        //获得内容区域根视图
        //activityRootView = (ViewGroup)decorView.findViewById(android.R.id.content);
        //Log.d("activityRootView count", activityRootView.getChildCount()+"") ;

        //获得我的根视图
        dialogView = (ViewGroup)layoutInflater.inflate(R.layout.dialog_progress,null);
        //Log.d("rootView count", rootView.getChildCount() + "") ;

        //屏蔽下层触摸
        dialogView.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                Log.d("lessonOneActivity","点击了本层");
            }
        });
        //屏幕返回键
        dialogView.setOnKeyListener(new View.OnKeyListener() {
            @Override
            public boolean onKey(View v, int keyCode, KeyEvent event) {
                return true;
            }
        });


    }
    public void show(){
        if(dialogView.getParent() == null){
            decorView.addView(dialogView);
            dialogView.setVisibility(View.VISIBLE);
        }
        else{
            //decorView.addView(rootView);
            dialogView.setVisibility(View.VISIBLE);
        }
    }

    public void dismiss(){
        dialogView.setVisibility(View.GONE);
    }

}

还是发现了上个泄漏问题,分析到这里,我已经没辙了,然后换台手机,魅族mx3,神奇的事情发生了,竟然没有泄漏了。黑人问号脸。

总结

继续查资料,发现很多人给出了讨论或者解决方案,倾向于Android输入法的漏洞,在15<=API<=23中都存在。
知乎用户十三太饱给出的解释是:
**
InputMethodManager的相关对象(mServedView等)没有传递下去的话,通过工具的检测的确会发现前一个Activity出现内存泄漏。但是实际上,InputMethodManager对象并不是完全归前一个Activity持有,只是暂时性的指向了它,InputMethodManager的对象是被整个APP循环的使用。另外,InputMethodManager是通过单例实现的,不会造成内存的叠加,所以个人觉得InputMethodManager并不会造成实质的内存泄漏。
**
个人选择不再解决,下面列一些blog供大家研究,有什么问题可以随时讨论,以上。

Android InputMethodManager 导致的内存泄露及解决方案
Leakcanary部分泄露警报无需修复

待研究参考资料:

Android非UI线程使用View.post()方法一处潜在的内存泄漏

注意事项:

  1. Dialog销毁一定要在activity销毁之前
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,922评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,591评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,546评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,467评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,553评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,580评论 1 293
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,588评论 3 414
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,334评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,780评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,092评论 2 330
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,270评论 1 344
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,925评论 5 338
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,573评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,194评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,437评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,154评论 2 366
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,127评论 2 352

推荐阅读更多精彩内容

  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 172,005评论 25 707
  • 介绍自己负责的部分,如何实现的。 自定义view viewGroup activity的启动流程 事件传递及滑动冲...
    东经315度阅读 1,206评论 1 4
  • 一、概述 Activity 作为与用户交互的一个窗口,是使用非常频繁的一个基本组件。Android系统是通过Act...
    三也视界阅读 2,233评论 3 11
  • 七年是一个轮回 早先的细胞都已作废 闲来身上不再爬满怨怼 可依旧没有把握 若再见你是否可以 自持不流泪 有时愿你我...
    Serena_02fc阅读 292评论 0 0
  • 【贱】多年不见之人,偶尔电话呼来,甚是欢欣;仰望的领导不经意看了你一眼,就想入非非;陪伴多年的夫妻,一切视而不见,...
    茧破阅读 497评论 8 11