Android 中的Context 上下文

原创 2016年05月31日 21:06:27

1、Context概念

其实一直想写一篇关于Context的文章,但是又怕技术不如而误人子弟,于是参考了些资料,今天准备整理下写出来,如有不足,请指出,参考资料会在醒目地方标明。

Context,相信不管是第一天开发Android,还是开发Android的各种老鸟,对于Context的使用一定不陌生~~你在加载资源、启动一个新的Activity、获取系统服务、获取内部文件(夹)路径、创建View操作时等都需要Context的参与,可见Context的常见性。大家可能会问到底什么是Context,Context字面意思上下文,或者叫做场景,也就是用户与操作系统操作的一个过程,比如你打电话,场景包括电话程序对应的界面,以及隐藏在背后的数据;

但是在程序的角度Context又是什么呢?在程序的角度,我们可以有比较权威的答案,Context是个抽象类,我们可以直接通过看其类结构来说明答案:


可以看到Activity、Service、Application都是Context的子类;

也就是说,Android系统的角度来理解:Context是一个场景,代表与操作系统的交互的一种过程。从程序的角度上来理解:Context是个抽象类,而Activity、Service、Application等都是该类的一个实现。

在仔细看一下上图:Activity、Service、Application都是继承自ContextWrapper,而ContextWrapper内部会包含一个base context,由这个base context去实现了绝大多数的方法。

先扯这么多,有能力了会从别的角度去审视Context,加油~

2、Context与ApplicationContext

看了标题,千万不要被误解,ApplicationContext并没有这个类,其实更应该叫做:Activity与Application在作为Context时的区别。嗯,的确是这样的,大家在需要Context的时候,如果是在Activity中,大多直接传个this,当在匿名内部类的时候,因为this不能用,需要写XXXActivity.this,很多哥们会偷懒,直接就来个getApplicationContext。那么大家有没有想过,XXXActivity.this和getApplicationContext的区别呢?

XXXActivity和getApplicationContext返回的肯定不是一个对象,一个是当前Activity的实例,一个是项目的Application的实例。既然区别这么明显,那么各自的使用场景肯定不同,乱使用可能会带来一些问题。

下面开始介绍在使用Context时,需要注意的问题。


3、引用的保持

大家在编写一些类时,例如工具类,可能会编写成单例的方式,这些工具类大多需要去访问资源,也就说需要Context的参与。

在这样的情况下,就需要注意Context的引用问题。

例如以下的写法:

package com.mooc.shader.roundimageview;  
  
import android.content.Context;  
  
public class CustomManager  
{  
    private static CustomManager sInstance;  
    private Context mContext;  
  
    private CustomManager(Context context)  
    {  
        this.mContext = context;  
    }  
  
    public static synchronized CustomManager getInstance(Context context)  
    {  
        if (sInstance == null)  
        {  
            sInstance = new CustomManager(context);  
        }  
        return sInstance;  
    }  
      
    //some methods   
    private void someOtherMethodNeedContext()  
    {  
          
    }  
}  
对于上述的单例,大家应该都不陌生(请别计较getInstance的效率问题),内部保持了一个Context的引用;

这么写是没有问题的,问题在于,这个Context哪来的我们不能确定,很大的可能性,你在某个Activity里面为了方便,直接传了个this;这样问题就来了,我们的这个类中的sInstance是一个static且强引用的,在其内部引用了一个Activity作为Context,也就是说,我们的这个Activity只要我们的项目活着,就没有办法进行内存回收。而我们的Activity的生命周期肯定没这么长,所以造成了内存泄漏。

那么,我们如何才能避免这样的问题呢?

有人会说,我们可以软引用,嗯,软引用,假如被回收了,你不怕NullPointException么。

把上述代码做下修改:

public static synchronized CustomManager getInstance(Context context)  
    {  
        if (sInstance == null)  
        {  
            sInstance = new CustomManager(context.getApplicationContext());  
        }  
        return sInstance;  
    }  
这样,我们就解决了内存泄漏的问题,因为我们引用的是一个ApplicationContext,它的生命周期和我们的单例对象一致。

这样的话,可能有人会说,早说嘛,那我们以后都这么用不就行了,很遗憾的说,不行。上面我们已经说过,Context和Application Context的区别是很大的,也就是说,他们的应用场景(你也可以认为是能力)是不同的,并非所有Activity为Context的场景,Application Context都能搞定。

下面就开始介绍各种Context的应用场景。


大家注意看到有一些NO上添加了一些数字,其实这些从能力上来说是YES,但是为什么说是NO呢?下面一个一个解释:

数字1:启动Activity在这些类中是可以的,但是需要创建一个新的task。一般情况不推荐。

数字2:在这些类中去layout inflate是合法的,但是会使用系统默认的主题样式,如果你自定义了某些样式可能不会被使用。

数字3:在receiver为null时允许,在4.2或以上的版本中,用于获取黏性广播的当前值。(可以无视)

注:ContentProvider、BroadcastReceiver之所以在上述表格中,是因为在其内部方法中都有一个context用于使用。


好了,这里我们看下表格,重点看Activity和Application,可以看到,和UI相关的方法基本都不建议或者不可使用Application,并且,前三个操作基本不可能在Application中出现。实际上,只要把握住一点,凡是跟UI相关的,都应该使用Activity做为Context来处理;其他的一些操作,Service,Activity,Application等实例都可以,当然了,注意Context引用的持有,防止内存泄漏。


5、总结

所以说:Android  Context上下文可这样使用:

1.凡是和UI相关的使用Activity(使用Application可能报错):例如:Start Activity;Layout Inflate;Dialog;Toast;

2.其他的一些操作,Service,Activity,Application等实例都可以,注意Context引用的持有,防止内存泄漏。

转载出处:鸿翔大神

Android Context 上下文 你必须知道的一切

版权声明:本文为博主原创文章,未经博主允许不得转载。

Android Context 上下文 你必须知道的一切

转载请标明出处:http://blog.csdn.net/lmj623565791/article/details/40481055,本文出自:【张鸿洋的博客】本文大多数内容翻译自:http://ww...
  • lmj623565791
  • lmj623565791
  • 2015年01月05日 09:21
  • 162884

Android: 如何在Fragment中使用上下文Context

时间 2015-08-11 15:34:55  dss886的博客 原文  http://www.dss886.com/android/2015/08/11/01/ 主题 安卓开发 ...
  • qq_23621841
  • qq_23621841
  • 2015年08月11日 18:17
  • 8262

Context全解析,让你的上下文明明白白

Activity mActivity =new Activity() 作为Android开发者,不知道你有没有思考过这个问题,Activity可以new吗?Android的应用程序开发采用JAV...
  • hello_word2
  • hello_word2
  • 2016年08月15日 16:05
  • 4660

单例模式之传递Context参数

单利模式的懒汉模式,在android的源码中应用的还是相当普遍的。在android的开发中,有时候需要一个Context的变量,同时还需要使用单例模式,那么该怎么处理呢? 在getInstance的时...
  • u013377887
  • u013377887
  • 2017年01月23日 16:01
  • 1237

程序中上下文Context

之前学习Toast的时候关于Toast.makeText()方法的第一个参数做了简单说明,现在详细说明一下这个Context。 Context是一个抽象基类,通过它访问当前包的资源(getRes...
  • zhangbohun
  • zhangbohun
  • 2016年02月25日 14:23
  • 778

Android获取其他包的Context,反射实现方法调用

Android中有Context的概念,想必大家都知道。Context可以做很多事情,打开activity、发送广播、打开本包下文件夹和数据库、获取classLoader、获取资源等等。如果我们得到了...
  • zhongwcool
  • zhongwcool
  • 2014年04月03日 11:10
  • 1292

React学习之进阶上下文(十九)

在React中很容易根据组件从而跟踪数据流,当你使用一个组件时,可以通过看props来判断用了哪一些东西。有时候你想直接通过组件树来讲数据传递子元素,而不是通过props一层一层的传递下去,可以使用R...
  • qq_18661257
  • qq_18661257
  • 2017年03月20日 21:21
  • 814

编程中什么是「Context(上下文)」?

参考: 1.https://www.zhihu.com/question/26387327 2.http://stackoverflow.com/questions/6145091/the-ter...
  • zhouhuakang
  • zhouhuakang
  • 2016年02月02日 11:58
  • 924

java上下文Context类

Context在Java中的出现是如此频繁,但其中文翻译“上下文”又是如此诡异拗口,因此导致很多人不是很了解Context的具体含义是指什么,所以很有必要来深究一下这词的含义。 先来举几个JAV...
  • baoendemao
  • baoendemao
  • 2014年05月05日 19:48
  • 2801

scala中的上下文绑定(context bound)

context boundimplicitlyimplicitly 主要是在当前作用域查找指定类型:def implicitly[T](implicit e : T) : T 例子:implicit ...
  • cjuexuan
  • cjuexuan
  • 2016年02月26日 21:52
  • 1197
内容举报
返回顶部
收藏助手
不良信息举报
您举报文章:Android 中的Context 上下文
举报原因:
原因补充:

(最多只允许输入30个字)