纯英文含义来看,Context 意指上下文、环境、背景等等……那么 Android 中的 Context 的含义和这些英文释义有什么联系呢?不妨看看 Google 给出的定义:
Interface to global information about an application environment. This is an abstract class whose implementation is provided by the Android system. It allows access to application-specific resources and classes, as well as up-calls for application-level operations such as launching activities, broadcasting and receiving intents, etc.
把它完全翻译下来我觉得没啥必要,那这段话的核心含义是啥呢:
回顾我们使用 Context 的场景来帮助理解吧:我们在使用自定义 View 时,使用 BaseAdapter 时,甚至是访问数据库文件时,都需要传入一个 Context 参数,大家有没有想过这到底是为什么呢?因为我们初始化自定义 View 需要将 View 与某个页面布局关联,因为我们使用 BaseAdapter 时需要某个布局文件作为子 Item,因为我们需要访问应用的数据库文件。此时 Context 就像一个系统信息管理员,你告诉它我想要访问系统的布局资源文件,想要访问应用的数据库文件,它就去给你找,然后提供引用给你使用。
所以大家现在应该能理解了吧,Context 就是用于获得系统资源的,我们所说的系统资源包括设备本身的信息,也有我们开发者提供的信息(类、包、资源文件等等……)。
在进行分析之前,我们不妨先看看 Context 的类继承图,了解 Context 的主要实现类有哪些:
如图中所见,Application 类和 Activity、Service 两大 Android 组件都是 Context 的具体实现类,而这三个类确实能访问各自职责内的系统资源,例如 Activity 能访问与界面元素相关的资源、Service 能访问系统服务、Application 能访问应用包名等信息。
知识点:应用中 Context 的数量 = Activity 的数量 + Service 的数量 + 1(Application 的数量)
前面已经提到,Context 的实现类能够持有各种各样的系统信息,可能有人会想到:使用 Context 不会导致内存泄漏吗?我在这里给大家答案:会!而且我们在开发中经常无意地导致了内存泄漏。有人可能就不服了,我写的代码怎么可能有内存泄漏的问题!
别急,我下面给大家看一段最普通的代码:
例如我们要实现一个 ListView 展示信息的功能,为了自定义子 Item,我们需要传入 Context 来读取布局资源:
public class MyBaseAdapter extends BaseAdapter{
private Context mContext;
public MyBaseAdapter(Context context) {
mContext = context;
}
省略……
}
设计完 BaseAdapter 以后,我们肯定要在 MainActivity 里面使用它了:
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
adapter = new MyBaseAdapter(this);
……省略
}
一般大家都会写下这样的代码对吧?这样写都不会出现 Bug 对吧?老板还夸你工作效率高对吧?事实上这样的代码会导致内存泄漏哦~为什么呢?大家不妨回顾我在在Activity中使用Thread导致的内存泄漏里提到的内存泄漏原因,博文中说道:只要类持有对外部类实例对象的引用,垃圾回收机制就不会回收该对象。
在这段代码中,我们将 Activity 的引用传入 MyBaseAdapter,使得 MyBaseAdapter 持有对 Activity 的引用,那么相应地,我们也将持有 Activity 所获得的资源文件的引用。所以这些资源文件将无法被垃圾回收机制回收,造成内存泄漏的问题。
那么有什么解决办法呢?
我们只需要修改一句代码:
mContext = context.getApplicationContext();
或者
adapter = new MyBaseAdapter(this.getApplication());
就可以避免内存泄漏的问题。大家需要记住的是,在这里我们只是演示了一种最常见,最基础的 Context 导致的内存泄漏问题,在实际的开发需求中肯定还有更加复杂的情况,大家应该更多地去思考背后的原因,从根源上解决这些内存泄漏的问题。
原文地址:http://blog.csdn.net/u012403246/article/details/46285151