Android开发——告诉你Adapter应该写在Activity里面还是外面

0. 前言

本文转载自AItsuki的博客。

首先说明一下为什么要写这么一篇博客:最近看了一些其他人的项目,发现很多项目的做法是建立一个专门存放Adapter类的Package包,也有的项目干脆直接都写在Activity中。而我属于后者,我并不觉得Adapter需要单独占用一个包,因为大多数时候他们并不能给其他Activity复用。

其实这两种方式都是可以的,但有一点很关键,那就是无论如何,adapter不能持有activity的引用,否则可能会因为adapter里面可能会做一些耗时操作,当activity finish时会因为被adapter持有引用而导致activity无法被回收,从而导致内存泄漏

 

1. 写在Activity外面也可能会导致内存泄漏

Java中,非静态内部类对象是会隐式持有外部类引用的,也就是说adapter持有了activity的引用。所以就有将Adapter写在单独的一个包中的做法,估计很多开发者都认为这样是为了让Activity看起来更简洁。

然而我还想说的是,adapter写在外面也阻止不了MDZZ程序员想在adapter内持有Activity引用。因为很多时候adapter不可避免的需要和Activity交互,或者需要一个context对象

 

比如说点击列表上的Item实现到跳转SecondActivity的效果,很多人可能这么做:

//外面的Adapter类
public class ExampleAdapter extends BaseAdapter {
    private Context mContext;
    public ExampleAdapter(Context context) {
        this.mContext = context;
    }

    @Override
    public View getView(int position, View convertView, ViewGroup parent) {
        if (convertView == null) {
            //这里也用到了activity的引用
            convertView = LayoutInflater.from(mContext).inflate(R.layout.list_item, null);
        }
        TextView textView = (TextView) convertView.findViewById(R.id.text);
        textView.setText(String.valueOf(position));

        Button button = (Button) convertView.findViewById(R.id.button);
        button.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                //使用构造方法传过来的上下文跳转界面
                mContext.startActivity(new Intent(mContext, SecondActivity.class));
            }
        });

        return convertView;
}


//MainActivity代码展示
public class MainActivity extends Activity {
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        ListView listView = (ListView) findViewById(R.id.listView);
        //这里将activity对象传入adapter
        ExampleAdapter adapter = new ExampleAdapter(this);
        listView.setAdapter(adapter);
    }
}

说实话,我第一次看上面代码,感觉很干净利落,甚至觉得有点舒服,==

但是这种情况下,adapter就持有了Activity引用,如果我们在adapter中进行耗时操作(比如加载图片之类的),并且finish Activity,虽然此时Activity销毁了,但是GC是无法回收activity,而且Activity占用的内存还是比较大的,这样就发生了内存泄漏。


2. 不持有Activity对象的情况下怎么和Activity交互

那么在上面的例子中,问题就集中在了如何在不持有Activity对象的情况下和Activity交互

 

2.1 首先处理getView()中的inflate()

当我们inflate一个xml时,完全可以使用parentcontext,实现如下:

 

2.2 点击事件,可以用回调接口

使用回调接口的方式来实现不持有activity的情况下,与Activity愉快的交互,实现如下:

 



3. 其实写在里面也可以呀

adapter写在Activity里面的话只需要加个static关键字(变为静态内部类)就行了,其他和写在外面是一样的。



至此关于关于ListView的Adapter应该写在Activity外面还是里面的问题就讨论结束了,其实两种方式都可以,只要留意内存泄漏的风险即可。


原文地址:https://www.cnblogs.com/qitian1/p/6461501.html