繁体   English   中英

Google-guava checkNotNull 和 IntelliJ IDEA 的“可能会产生 java.lang.NullPointerException”

[英]Google-guava checkNotNull and IntelliJ IDEA's “may produce java.lang.NullPointerException”

有没有办法抑制这个警告:

MyClass object = null;

/*Some code that 'might' set this object but I know it will*/      


Preconditions.checkNotNull(object); 
//when "assert object != null" is used here no warning is shown

merged.setName(dRElement.getName());
//"May produce 'java.lang.NullPointerException'" warning here 

我正在使用 IntelliJ IDEA 10.5,我知道这个警告是不必要的,但是我想在这里压制它并避免关闭检查。

通过@Contract注释和外部注释功能的组合,您现在可以注释Preconditions方法,以便 IntelliJ 将正确的 static 分析应用于对这些方法的调用。

假设我们有这个例子

public void doSomething(Object someArg) {
    Preconditions.checkArgument(someArg != null);
    someArg.doSomethingElse();  //currently gives NPE warning

    if (someArg != null) {
        //no warning that this is always true
    }
}

在 IntelliJ 中(我使用的是 13):

  • 导航到Preconditions.checkArgument(boolean)
  • 将 cursor 放在方法名称上,然后按 Alt - Enter以弹出意图弹出窗口。
  • Select“添加方法合同”。
  • 使用合同文本false -> fail
  • 出现提示时,提供外部注释文件的位置。

现在someArg.doSomethingElse()的警告消失了,IDEA 实际上会将if分支标记为始终为真!

其他合同文本:

  • Preconditions.checkArgument(boolean, String)应该为false, _ -> fail
  • Preconditions.checkNotNull(Object, String)应该是null, _ -> fail ,
  • 等等等等

这是我的完整annotations.xml文件的Preconditions

<root>
    <item name='com.google.common.base.Preconditions T checkNotNull(T)'>
        <annotation name='org.jetbrains.annotations.Contract'>
            <val val="&quot;null -&gt; fail&quot;"/>
        </annotation>
    </item>
    <item name='com.google.common.base.Preconditions T checkNotNull(T, java.lang.Object)'>
        <annotation name='org.jetbrains.annotations.Contract'>
            <val val="&quot;null, _ -&gt; fail&quot;"/>
        </annotation>
    </item>
    <item name='com.google.common.base.Preconditions T checkNotNull(T, java.lang.String, java.lang.Object...)'>
        <annotation name='org.jetbrains.annotations.Contract'>
            <val val="&quot;null, _, _ -&gt; fail&quot;"/>
        </annotation>
    </item>
    <item name='com.google.common.base.Preconditions void checkArgument(boolean)'>
        <annotation name='org.jetbrains.annotations.Contract'>
            <val val="&quot;false -&gt; fail&quot;"/>
        </annotation>
    </item>
    <item name='com.google.common.base.Preconditions void checkArgument(boolean, java.lang.Object)'>
        <annotation name='org.jetbrains.annotations.Contract'>
            <val val="&quot;false, _ -&gt; fail&quot;"/>
        </annotation>
    </item>
    <item name='com.google.common.base.Preconditions void checkArgument(boolean, java.lang.String, java.lang.Object...)'>
        <annotation name='org.jetbrains.annotations.Contract'>
            <val val="&quot;false, _, _ -&gt; fail&quot;"/>
        </annotation>
    </item>
    <item name='com.google.common.base.Preconditions void checkState(boolean)'>
        <annotation name='org.jetbrains.annotations.Contract'>
            <val val="&quot;false -&gt; fail&quot;"/>
        </annotation>
    </item>
    <item name='com.google.common.base.Preconditions void checkState(boolean, java.lang.Object)'>
        <annotation name='org.jetbrains.annotations.Contract'>
            <val val="&quot;false, _ -&gt; fail&quot;"/>
        </annotation>
    </item>
    <item name='com.google.common.base.Preconditions void checkState(boolean, java.lang.String, java.lang.Object...)'>
        <annotation name='org.jetbrains.annotations.Contract'>
            <val val="&quot;false, _, _ -&gt; fail&quot;"/>
        </annotation>
    </item>
</root>

也可以看看

JetBrains Yourtrack 中存在添加此类功能的老问题 几年前我投票支持它,但我没有看到任何活动。 如果每个人都投票支持它,那么我们可能会很幸运。

澄清问题包括添加功能,以便您可以将方法标记为执行某种类型的 null 检查。 如果发生这种情况,那么您可以为 Preconditions 方法编写自己的包装器并对其进行注释。

更新我厌倦了等待功能,所以我自己提交了一个补丁。 它在 12.1.1 版本 129.239 中可用。 要访问配置:设置 > 检查 > 可能的错误 > 常量条件和异常 > 配置断言/检查方法。

提取方法?

private MyClass getMyClass() {
    /* This always returns an instance of MyClass, never null. */      
}

...

MyClass object = getMyClass();
Preconditions.checkNotNull(object);
merged.setName(object.getName());

您还可以从 checkNotNull 方法分配返回值,该值非空: http://docs.guava-libraries.googlecode.com/git-history/release/javadoc/com/google/common/base/Preconditions.html# checkNotNull(T)

MyClass object = null;
object = Preconditions.checkNotNull(object); 
merged.setName(dRElement.getName());

从 IntelliJ IDEA 14 开始,您无需担心这一点。

@Nullable
String extractPrefix(@Nullable String url) {
    if (StringUtils.isEmpty(url)) return null;

    if (url.startsWith("jar://")) {
...

早期的 IntelliJ IDEA 不知道如果 url 是 null,代码执行甚至不会到达“startsWith”调用,因为它没有查看 StringUtils.isEmpty 内部。 您当然可以通过自己查看 isEmpty 并添加合同“null -> true”来删除 extractPrefix 中的那些黄色警告,但这太无聊了,这正是计算机的工作,而不是你,现在 IntelliJ IDEA 会自动完成。 查看字节和源代码。

http://blog.jetbrains.com/idea/2014/10/automatic-notnullablecontract-inference-in-intellij-idea-14/

除了其他选择,您可以尝试-

通过使用 - 在方法级别抑制警告 -

@SuppressWarnings({"NullableProblems"})
public void someMethod(){
   ...
   ...
}

或通过使用注释来抑制语句级别的警告 -

//noinspection NullableProblems
someMethodCallProducingNullWarning(null);

但在这样做之前 - 确保它不会真正产生 NPE

暂无
暂无

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM