Java到Kotlin转换器和可空的方法参数

时间:2017-10-09 23:25:39

标签: java android kotlin kotlin-android-extensions kotlin-null-safety

我有一个案例,其中Java to Kotlin转换器因为没有将方法参数标记为可空而使我失败。

示例:使用registerActivityLifecycleCallbacks跟踪活动生命周期:

registerActivityLifecycleCallbacks(new ActivityLifecycleCallbacks() {
    @Override
    public void onActivityCreated(Activity activity, Bundle savedInstanceState) {}

    @Override
    public void onActivityStarted(Activity activity) {}

    @Override
    public void onActivityResumed(Activity activity) {}

    // ... other overriden methods
});

将此代码粘贴到Kotlin结果:

registerActivityLifecycleCallbacks(object : Application.ActivityLifecycleCallbacks {
    override fun onActivityCreated(activity: Activity, savedInstanceState: Bundle) {}

    override fun onActivityStarted(activity: Activity) {}

    override fun onActivityResumed(activity: Activity) {}

    override fun onActivityPaused(activity: Activity) {}

    // ... other overriden methods (all with non-nullable parameters)
})

问题是savedInstanceState参数类型为Bundle,它应该是Bundle?,因为它的值可以是null

在这种情况下,当创建Activity而没有实例状态时,我们将得到以下异常:

java.lang.IllegalArgumentException: Parameter specified as non-null is null: method kotlin.jvm.internal.Intrinsics.checkParameterIsNotNull, parameter savedInstanceState

注意,导致这种情况的根本原因可能是onActivityCreated documentation 没有提及Bundle可以为空onCreate documentation可以解释为什么简单的onCreate转换按预期工作:

// Java
@Override
protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
}
// Kotlin
override fun onCreate(savedInstanceState: Bundle?) {
    super.onCreate(savedInstanceState)
    setContentView(R.layout.activity_main)
}

我的问题是我们如何知道哪些论据可以为可以阻止这些问题? @Nullable 注释在这里没有帮助。

1 个答案:

答案 0 :(得分:4)

如果注释无效,我认为没有办法知道参数是否可以为空。

关于您发布的代码,我想我知道发生了什么:

活动onCreate IS 标记为@Nullable

@Override
protected void onCreate(@Nullable Bundle savedInstanceState) {
    final AppCompatDelegate delegate = getDelegate();
    delegate.installViewFactory();
    delegate.onCreate(savedInstanceState);
    .
    .

虽然ActivityLifecycleCallbacks界面方法不是 :(参见Application.java

public interface ActivityLifecycleCallbacks {
    void onActivityCreated(Activity activity, Bundle savedInstanceState);
    void onActivityStarted(Activity activity);
    void onActivityResumed(Activity activity);
    void onActivityPaused(Activity activity);
    void onActivityStopped(Activity activity);
    void onActivitySaveInstanceState(Activity activity, Bundle outState);
    void onActivityDestroyed(Activity activity);
}

显然Kotlin翻译器正在处理注释,但是使用 Non-Null 作为默认值,这在我看来并不常见,因为对于非注释参数更有意义< EM>可空。但是,我可以理解这个决定,以迫使开发人员关注已翻译的代码并明确决定参数是否为 Nullable

顺便说一下,请注意有几个@NonNull@Nullable注释(javax,android,jetbrains ......)如果Kotlin翻译只识别其中一些(但是这只是一个猜测)

另外关于你的代码,Java Lint应该给你一个关于你被覆盖的onCreate的警告,说你要覆盖一个带有注释参数的方法而你没有注释它们。