博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
Android中Context解析
阅读量:6618 次
发布时间:2019-06-25

本文共 3987 字,大约阅读时间需要 13 分钟。

Context概念

当我们访问当前应用的资源,启动一个新的activity的时候都需要提供Context。

Context是一个抽象基类,我们通过它访问当前包的资源(getResources、getAssets)和启动其他组件(Activity、Service、Broadcast)以及得到各种服务(getSystemService),当然,通过Context能得到的不仅仅只有上述这些内容。对Context的理解可以来说:Context提供了一个应用的运行环境,在Context的大环境里,应用才可以访问资源,才能完成和其他组件、服务的交互,Context定义了一套基本的功能接口,可以理解为一套规范,而Activity和Service是实现这套规范的子类,这么说也许并不准确,因为这套规范实际是被ContextImpl类统一实现的,Activity和Service只是继承并有选择性地重写了某些规范的实现。

 

 

activity继承关系:

  

Service和Application:

 

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

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

Context的继承结构还是稍微有点复杂的,可以看到,直系子类有两个,一个是ContextWrapper,一个是ContextImpl。那么从名字上就可以看出,ContextWrapper是上下文功能的封装类,而ContextImpl则是上下文功能的实现类。而ContextWrapper又有三个直接的子类,ContextThemeWrapper、Service和Application。其中,ContextThemeWrapper是一个带主题的封装类,而它有一个直接子类就是Activity。

那么在这里我们至少看到了几个所比较熟悉的面孔,Activity、Service、还有Application。由此,其实我们就已经可以得出结论了,Context一共有三种类型,分别是Application、Activity和Service。这三个类虽然分别各种承担着不同的作用,但它们都属于Context的一种,而它们具体Context的功能则是由ContextImpl类去实现的。

那么Context到底可以实现哪些功能呢?这个就实在是太多了,弹出Toast、启动Activity、启动Service、发送广播、操作数据库等都需要用到Context。由于Context的具体能力是由ContextImpl类去实现的,因此在绝大多数场景下,Activity、Service和Application这三种类型的Context都是可以通用的。不过有几种场景比较特殊,比如启动Activity,还有弹出Dialog。出于安全原因的考虑,Android是不允许Activity或Dialog凭空出现的,一个Activity的启动必须要建立在另一个Activity的基础之上,也就是以此形成的返回栈。而Dialog则必须在一个Activity上面弹出(除非是System Alert类型的Dialog),因此在这种场景下,我们只能使用Activity类型的Context,否则将会出错。

一个应用中Context的数量等于Activity的个数 + Service的个数 + 1,这个1为Application

 

Context与ApplicationContext

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

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

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

 

引用的保持

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

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

例如以下的写法:

 

[java]   
 
 
  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]   
 
 
  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的应用场景。

 

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引用的持有,防止内存泄漏。

转载于:https://www.cnblogs.com/weiguang/p/6577204.html

你可能感兴趣的文章
oracle使用dba用户导出数据,Oracle创造dba用户+导入导出数据库表
查看>>
linux下attrd进程,【Linux】第二章 Linux进程与线程(下)
查看>>
linux 内核配置 fatfs,微软发布exFAT文件系统规范:推动嵌入Linux内核 支持开箱即用...
查看>>
linux程序中文乱码转换,Linux下汉字编码的转换(gbk转换为utf8)
查看>>
linux tty终端 录屏,在Linux上录制终端的操作
查看>>
linux i3 桌面,Linux 桌面平铺管理器 - i3wm
查看>>
linux文件内容转置,给定一个文件 file.txt,转置它的内容。你可以假设每行列数相同,并且每个字段由 ' ' 分隔....
查看>>
linux下mysql一般安装在哪里,MySQL在Linux下的安装
查看>>
linux malloc 多线程,c-malloc如何在多线程环境中工作?
查看>>
linux系统串口使用,linux使用串口说明
查看>>
linux实验串口网络通信编程仪,Linux串口通信编程
查看>>
linux下怎么绑定arp,LINUX 下进行arp 绑定.doc
查看>>
sys_brk分析 linux1.2.0版本,linux内存管理之sys_brk实现分析(续)
查看>>
c语言14年春机考答案,计算机二级《C语言》机考试题与答案
查看>>
c语言括号匹配问题不用栈,括号匹配问题(不用栈,用数组)
查看>>
c语言指针课堂教学设计,“C语言程序设计”课程指针的教学设计
查看>>
gradle是否可以编译c语言,使用CPP插件在gradle中编译C++代码
查看>>
c语言程实训报告,C语言程设计工程实训报告.doc
查看>>
C语言高斯消元课程设计报告,c高斯消元法解方程-课程设计报告.doc
查看>>
三坐标DMIS语言是C语言吗,三坐标测量软件PC-DMIS常见技巧
查看>>