为什么有时候启动Activity需要加FLAG_ACTIVITY_NEW_TASK

问题描述:

我们通过广播来启动Activity的时候如果不设置intent的FLAG_ACTIVITY_NEW_TASK属性,就会报这个异常:

android.util.AndroidRuntimeException: Calling startActivity() from outside of an Activity  context requires the FLAG_ACTIVITY_NEW_TASK flag.

就是说在activity上下文之外(即除了activity调用的之外)调用startActivity需要FLAG_ACTIVITY_NEW_TASK属性。

问题分析:

  1. Context是什么:Context描述的是一个应用程序环境的信息,这是一个抽象(abstract class)类,Android提供了该抽象类的很多具体实现类,比如我们常用的Service,Application,Activity。通过它我们可以获取应用程序的资源和类,也包括一些应用级别操作,例如:启动一个Activity,发送广播,接受Intent信息等。

图一 (图片内容来自http://blog.csdn.net/qinjuning/article/details/7310620)

  1. Activity上下文之外是什么意思:通过debug发现在Receiver->onReceive的时候传进来的context是ReceiverRestrictedContext,然而ReceiverRestrictedContext的代码很简单,里面没有startActivity方法,而ReceiverRestrictedContext是继承自ContextWrapper,ContextWrapper的startActivity方法如下:

     

    [java]  view plain  copy
    1. public void startActivity(Intent intent, Bundle options) {  
    2.     warnIfCallingFromSystemProcess();  
    3.     if((intent.getFlags() & Intent.FLAG_ACTIVITY_NEW_TASK) == 0) {  
    4.         thrownew AndroidRuntimeException(  
    5.                 "Calling startActivity() from outside of an Activity "  
    6.                         +" context requires the FLAG_ACTIVITY_NEW_TASK flag."  
    7.                         +" Is this really what you want?");  
    8.     }  
    9.     mMainThread.getInstrumentation().execStartActivity(  
    10.             getOuterContext(), mMainThread.getApplicationThread(), null,  
    11.             (Activity)null, intent, -1, options);  
    12. }  

    可以看到如果我们在Receiver中使用context的startActivity方法的话,这个方法会在执行真正的调用之前会检查一下有没有设置这个FLAG_ACTIVITY_NEW_TASK的标志,没有设置的话就报上面所说的那个异常。那么为什么我们在Activity中直接startActivity方法就不会报这个异常呢?这是因为Activity类在自己的实现中已经覆盖了父类的startActivity方法去除了这个检查。至于这个标志最终用在哪里我还没有跟踪到,因为最终启动activity调用的是native方法,如果后面有时间研究这一块我再补充上来。下面第三点主要从表层结合谷歌的文档分析一下FLAG_ACTIVITY_NEW_TASK到底有什么作用。

    [java]  view plain  copy
    1. @Override  
    2. publicvoid startActivity(Intent intent) {  
    3.     mBase.startActivity(intent);  
    4. }  

    可已看到,实际上调用的是mBase的startActivity方法,通过图一可以看到,mBase的基础实现类就是ContextImpl类,于是定位到ContextImpl的startActivity方法中:
  2. 浅谈设置FLAG_ACTIVITY_NEW_TASK的意义:首先需要了解一下什么是任务堆栈,大家可以直接看谷歌的文档,这里已经解释的很清楚了:打开链接。那么FLAG_ACTIVITY_NEW_TASK的意义可以这么理解,一般来说当我们从launcher中启动一个应用进入到ActivityA中,系统会为这个应用生成一个新任务堆栈并置于前台,ActivityA被放入栈底,之后从ActivityA启动另一个ActivityB,如果不设置什么附加属性,ActivityB默认也放到和ActivityA这个堆栈中,这样当你按返回时,B出栈,A呈现出来了,这个应该很好理解。那现在假如ActivityA启动一个Service或者发一个广播,这都是后台的活,和我们的任务栈没有关系,现在假如我们在广播中需要启动一个Activity,当然需要为这个Activity指定或分配一个任务栈,FLAG_ACTIVITY_NEW_TASK的意义就是这个,官方的文档解释为:“Start the activity in a new task. If a task is already running for the activity you are now starting, that task is brought to the foreground with its last state restored and the activity receives the new intent in onNewIntent().”

问题总结:

在Activity上下文之外启动Activity需要给Intent设置FLAG_ACTIVITY_NEW_TASK标志,不然会报异常。

猜你喜欢

转载自blog.csdn.net/u014748504/article/details/79805174