美文网首页Android开发经验谈程序员Android技术知识
每日一问:context.startActivity() 不设置

每日一问:context.startActivity() 不设置

作者: nanchen2251 | 来源:发表于2019-07-15 20:19 被阅读28次

    坚持原创日更,短平快的 Android 进阶系列,敬请直接在微信公众号搜索:nanchen,直接关注并设为星标,精彩不容错过。

    想必大多数人都知道在我们使用非 ActivitystartActivity() 的时候,都需要指定 Intent.FLAG_ACTIVITY_NEW_TASK,如果没有指定,直接进行操作则会直接抛出异常。

    上面我们使用 applicationContextstartActivity() 操作,不出意外的引发了崩溃,而正确的代码是:

    val intent = Intent(this, Main2Activity::class.java)
    intent.flags = Intent.FLAG_ACTIVITY_NEW_TASK
    applicationContext.startActivity(intent)
    

    这本身并没有什么值得争议的地方,但真的不加这个 FLAG,在手机上就一定会发生崩溃么?实际上,不加 FLAG 的处理也并不一定在手机上发生上述崩溃。

    applicationContext.startActivity(Intent(this, Main2Activity::class.java))
    

    上述的代码,有明显的问题,我们使用 applicationContext 来做 startActivity() 操作,却没有指定任何的 FLAG,但是,在 8.0 的手机上,你一定会惊讶的发现,我们并没有等到意料内的崩溃日志,而且跳转也是非常正常,这不由得和我们印象中必须加 FLAG 的结论大相径庭。然后再拿一个 9.0 的手机来尝试,马上就出现了上面的崩溃。

    这是为什么呢?我们必须看看源码。我们先基于 SDK 26,直接打开 Context 的实现类 ContextImpl,直接通过关键字 context requires the FLAG_ACTIVITY_NEW_TASK flag 搜索定位到下面的方法。

    当然,这里其实也可以直接一层一层跟进源码找到这个方法,效果一样。只是既然我们都通过日志知道了异常 message,那么直接通过异常关键字搜索一定是最快的,这个检索方法在很多时候非常有用!

    @Override
    public void startActivity(Intent intent, Bundle options) {
        warnIfCallingFromSystemProcess();
    
        // Calling start activity from outside an activity without FLAG_ACTIVITY_NEW_TASK is
        // generally not allowed, except if the caller specifies the task id the activity should
        // be launched in.
        if ((intent.getFlags()&Intent.FLAG_ACTIVITY_NEW_TASK) == 0
                && options != null && ActivityOptions.fromBundle(options).getLaunchTaskId() == -1) {
            throw new AndroidRuntimeException(
                    "Calling startActivity() from outside of an Activity "
                    + " context requires the FLAG_ACTIVITY_NEW_TASK flag."
                    + " Is this really what you want?");
        }
        mMainThread.getInstrumentation().execStartActivity(
                getOuterContext(), mMainThread.getApplicationThread(), null,
                (Activity) null, intent, -1, options);
    }   
    

    然后我们再基于 SDK 28 打开源码:

    @Override
    public void startActivity(Intent intent, Bundle options) {
        warnIfCallingFromSystemProcess();
    
        // Calling start activity from outside an activity without FLAG_ACTIVITY_NEW_TASK is
        // generally not allowed, except if the caller specifies the task id the activity should
        // be launched in. A bug was existed between N and O-MR1 which allowed this to work. We
        // maintain this for backwards compatibility.
        final int targetSdkVersion = getApplicationInfo().targetSdkVersion;
    
        if ((intent.getFlags() & Intent.FLAG_ACTIVITY_NEW_TASK) == 0
                && (targetSdkVersion < Build.VERSION_CODES.N
                        || targetSdkVersion >= Build.VERSION_CODES.P)
                && (options == null
                        || ActivityOptions.fromBundle(options).getLaunchTaskId() == -1)) {
            throw new AndroidRuntimeException(
                    "Calling startActivity() from outside of an Activity "
                            + " context requires the FLAG_ACTIVITY_NEW_TASK flag."
                            + " Is this really what you want?");
        }
        mMainThread.getInstrumentation().execStartActivity(
                getOuterContext(), mMainThread.getApplicationThread(), null,
                (Activity) null, intent, -1, options);
    }
    

    注释已经写的很清楚了,我们使用 Context.startActivity() 的时候是一定要加上 FLAG_ACTIVITY_NEW_TASK 的,但是在 Android N 到 O-MR1,即 24~27 之间却出现了 bug,即使没有加也会正确跳转。

    对比源码发现,在我们非 Activity 调用 startActivity() 的时候,我们这个 options 通常是 null 的,所以在 24~27 之间的时候,误把判断条件 options == null 写成了 options != null 导致进不去 if,从而不会抛出异常。

    关于启动模式的话,之前在面试系列已经对这一块知识点进行详细讲述,可直接点击链接前往:Android 面试:说说 Android 的四种启动模式

    相关文章

      网友评论

        本文标题:每日一问:context.startActivity() 不设置

        本文链接:https://www.haomeiwen.com/subject/vffokctx.html