我使用注释来确保参数不为空,假设这会导致编译器检查。
public @Nullable ApplicationAccount accountForKey(@NonNull String key) {
return accounts.get(key);
}
但是,通过运行这段代码,我在这一行得到了 NullPointerException
java.util.concurrent.ConcurrentHashMap.get (ConcurrentHashMap.java:883)
那么注释的重点是什么?
更晦涩的是,如果我像这样写额外的支票
return key!=null?accounts.get(key):null;
Android Studio 警告我支票没用!
更新:完整的调用堆栈:
Caused by java.lang.NullPointerException
at java.util.concurrent.ConcurrentHashMap.get(ConcurrentHashMap.java:883)
at co.getcontrol.services.MerchantCenter.accountForKey(MerchantCenter.java:72)
at co.getcontrol.model.customers.CustomersAggregator.loadCustomerDetails(CustomersAggregator.java:91)
at co.getcontrol.model.customers.CustomerDetailsPresenter.callData(CustomerDetailsPresenter.java:39)
at co.getcontrol.reskin.ui.customers.CustomerDetailsViewFragment.onCreateView(CustomerDetailsViewFragment.java:152)
at android.support.v4.app.Fragment.performCreateView(Fragment.java:1974)
at android.support.v4.app.FragmentManagerImpl.moveToState(FragmentManager.java:1067)
at android.support.v4.app.FragmentManagerImpl.moveToState(FragmentManager.java:1252)
at android.support.v4.app.BackStackRecord.run(BackStackRecord.java:738)
at android.support.v4.app.FragmentManagerImpl.execPendingActions(FragmentManager.java:1617)
at android.support.v4.app.FragmentController.execPendingActions(FragmentController.java:339)
at android.support.v4.app.FragmentActivity.onStart(FragmentActivity.java:602)
at co.getcontrol.ui.ControlActivity.onStart(ControlActivity.java:13)
at android.app.Instrumentation.callActivityOnStart(Instrumentation.java:1174)
at android.app.Activity.performStart(Activity.java:5353)
at android.app.ActivityThread.performLaunchActivity(ActivityThread.java:2352)
at android.app.ActivityThread.handleLaunchActivity(ActivityThread.java:2441)
at android.app.ActivityThread.access$900(ActivityThread.java:151)
at android.app.ActivityThread$H.handleMessage(ActivityThread.java:1354)
at android.os.Handler.dispatchMessage(Handler.java:110)
at android.os.Looper.loop(Looper.java:193)
at android.app.ActivityThread.main(ActivityThread.java:5345)
at java.lang.reflect.Method.invokeNative(Method.java)
at java.lang.reflect.Method.invoke(Method.java:515)
at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:828)
at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:644)
at dalvik.system.NativeStart.main(NativeStart.java)
最佳答案
注释创建契约(Contract)。 @NonNull
表示这个方法不接受 null
并且传递它可能会导致程序崩溃(这正是发生的事情)。 Android Studio 将警告任何使用此方法的地方,它可以推断出 null
可能会被传递。但它不会阻止传递 null
。
非常相似的契约是 []
的参数,用于不能超出其边界(例如小于 0)的数组,但这仍然不会阻止开发人员传递超出边界的值。null
的检查被标记为多余的,因为根据契约(Contract),永远不应该有 null
。如果添加检查,该方法现在将知道如何处理 null
,因此不应标记为 @NonNull
。