关闭

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

标签: AndroidContext上下文
229人阅读 评论(0) 收藏 举报
分类:

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的引用问题。

例如以下的写法:

[java] view plaincopy在CODE上查看代码片派生到我的代码片
  1. package com.mooc.shader.roundimageview;  
  2.   
  3. import android.content.Context;  
  4.   
  5. public class CustomManager  
  6. {  
  7.     private static CustomManager sInstance;  
  8.     private Context mContext;  
  9.   
  10.     private CustomManager(Context context)  
  11.     {  
  12.         this.mContext = context;  
  13.     }  
  14.   
  15.     public static synchronized CustomManager getInstance(Context context)  
  16.     {  
  17.         if (sInstance == null)  
  18.         {  
  19.             sInstance = new CustomManager(context);  
  20.         }  
  21.         return sInstance;  
  22.     }  
  23.       
  24.     //some methods   
  25.     private void someOtherMethodNeedContext()  
  26.     {  
  27.           
  28.     }  
  29. }  

对于上述的单例,大家应该都不陌生(请别计较getInstance的效率问题),内部保持了一个Context的引用;

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

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

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

把上述代码做下修改:

[java] view plaincopy在CODE上查看代码片派生到我的代码片
  1. public static synchronized CustomManager getInstance(Context context)  
  2.     {  
  3.         if (sInstance == null)  
  4.         {  
  5.             sInstance = new CustomManager(context.getApplicationContext());  
  6.         }  
  7.         return sInstance;  
  8.     }  

这样,我们就解决了内存泄漏的问题,因为我们引用的是一个ApplicationContext,它的生命周期和我们的单例对象一致。

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

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


4、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、总结

好了,到此,Context的分析基本完成了,希望大家在以后的使用过程中,能够稍微考虑下,这里使用Activity合适吗?会不会造成内存泄漏?这里传入Application work吗?

由于参考内容过多,本文改为译文咯~~


强烈推荐你再看这一篇关于Context的讲解http://blog.csdn.net/mrleeapple/article/details/49908703

转载出处:http://blog.csdn.net/lmj623565791/article/details/40481055,本文出自:【张鸿洋的博客

本文大多数内容翻译自:http://www.doubleencore.com/2013/06/context/  我重新组织了下内容以及结构,建议大家尽可能看下原文。


0
0
查看评论

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

原地址:http://blog.csdn.net/lmj623565791/article/details/40481055 转载请标明出处:http://blog.csdn.net/lmj623565791/article/details/40481055,本文出自:【张鸿洋的博客】...
  • duyiqun
  • duyiqun
  • 2017-01-16 21:59
  • 106

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

转载请标明出处:http://blog.csdn.net/lmj623565791/article/details/40481055,本文出自:【张鸿洋的博客】本文大多数内容翻译自:http://www.doubleencore.com/2013/06/context/  我重新...
  • lmj623565791
  • lmj623565791
  • 2015-01-05 09:21
  • 165787

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

转载请标明出处:http://blog.csdn.net/lmj623565791/article/details/40481055,本文出自:【张鸿洋的博客】 本文大多数内容翻译自:http://www.doubleencore.com/2013/06/context/  ...
  • u011494050
  • u011494050
  • 2015-01-05 16:31
  • 510

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

转载自:http://blog.csdn.net/lmj623565791/article/details/40481055,本文出自:【张鸿洋的博客】 本文大多数内容翻译自:http://www.doubleencore.com/2013/06/context/  我重新组织了...
  • IT1039871366
  • IT1039871366
  • 2015-02-28 15:14
  • 357

你必须知道的.NET(第2版)(奋斗的小鸟)_PDF 电子书

下载地址:http://pan.baidu.com/share/link?shareid=1270688138&uk=721744522
  • tjoy2005
  • tjoy2005
  • 2013-10-09 19:11
  • 2075

你必须知道的.NET(第2版)

王涛 编著 ISBN 978-7-121-14128-7   2011年7月出版 定价:79.00元 16开 548页 宣传语:以深入浅出的笔法,触摸.NET底层和框架的有趣角落,了解.NET平台的开发、设计与框架,剥丝抽茧,探微知著。从语言细节到设计架构,娓娓道来,
  • broadview2006
  • broadview2006
  • 2011-08-09 17:56
  • 4257

你需要知道的Android上下文Context

Context对于Android开发者来说肯定不陌生,在我们跳转新的Activity、弹出Toast,创建View等行为时都需要用到Context,可见Context每天都伴随着我们,但是Context究竟是什么意思呢?Context从中文翻译上是上下文、环境、场景。我个人更倾向用场景来理解Cont...
  • qq_435559203
  • qq_435559203
  • 2017-02-07 16:57
  • 345

你必须知道的.NET

主要讲.NET底层的一些语言机制,载体为C#。写得比较深入,是本好书。LOH堆 (Large Object Heap) ,用于分配大对象实例。LOH堆不会被压缩,而且只在完全GC回收时被回收 并非所有的值类型都创建在线程的堆栈上,当值类型作为类的字段、或作为堆上实例成员的一部分、或发生装箱时,该值...
  • s10141303
  • s10141303
  • 2015-07-03 23:30
  • 1595

数据挖掘 你必须知道的32个经典案例(电子书)

需要完整版电子书的加QQ:404178824
  • Siphiababy
  • Siphiababy
  • 2017-09-12 20:57
  • 574

Android中最常使用的Context上下文相关总结

其实在学习Android的时候,遇到上下文这个概念的时候,非常的迷糊,因为contex的种类太多了,什么getBaseContext(),  getApplicationContext()......或多种Context, 都不知道怎么使用Context.今天有空,坐下来学习一点新的东西,...
  • lanxingfeifei
  • lanxingfeifei
  • 2016-02-16 15:36
  • 1700
    个人资料
    • 访问:211257次
    • 积分:2909
    • 等级:
    • 排名:第14393名
    • 原创:24篇
    • 转载:356篇
    • 译文:2篇
    • 评论:4条
    最新评论