欢迎您访问第A百科网

null网络什么意思(烦人的NULL,你可以走开点了)

100次浏览     发布时间:2024-09-19 09:46:03     编辑: 狗都不学Java

1. Null 的问题

假设现在有一个需要三个参数的方法。其中第一个参数是必须的,后两个参数是可有可无的。

第一种情况,在我们调用这个方法的时候,我们只能传入两个参数,对第三个参数,我们在上下文里是没有的,那么我们调用方法的时候,就需要用一个特殊值去告知这个方法:
第三个参数我们拿不到,参数是不存在或者不明确的。
这个特殊的值应该用什么呢?在 Java 中,我们会选择用 null 去表示这种情况。

第二种情况,如果在调用方法的时候,我们有三个参数,只是第三个参数没有值,我们也需要传入一个特殊的值去表示:
参数存在,但是没有值。
这个特殊的值是什么呢?没错,在 Java 中,又是 null。

你看到了,现在 null 值的含义本身出现了两个意思:

  1. 参数不存在
  2. 参数没有值

二义性在计算机科学里是能避免就尽量避免的。所以,null 值的二义性是一个 Java 中的设计缺陷。不过,也不光是在 Java 语言中,null 的二义性在编程语言里是广泛存在的一个问题。这个问题被称为 Null 引用问题。

Null 引用是计算机科学中一个历史悠久又臭名昭著的问题。在 1964 年,由快排算法的创造者东尼·霍尔发明。他自称这是个十亿美元的错误。

在 Java 中,当我们去调用一个对象值为 null 的方法或者属性时,就会报
java.lang.NullPointerException,简称为 NPE。

传统上,这些 NPE 问题,必须完全依赖程序员本身细致周密的检查,对于 null 的检查充斥在了 Java 代码的字里行间,让代码变得臃肿丑陋,非常恶心。

同时,由于 NPE 的二义性问题,开发人员往往无法完全防护住 NPE,这使得 NPE 成为了开发人员的噩梦。明明逻辑上,一个对象是存在的,只是不知道其明确含义,但是只要引用了这个没有明确含义值的对象的方法,就会被告知NPE,简直让人防不胜防。

并且,更可恶的是,在 Java 中,NPE 是运行期异常,这就意味着 NPE 无法早期发现,只有上线运行了,才可能出现问题。

讨厌的 null,成本巨大的 NPE,让 Java 开发人员在不断地实践中,采用了各种方法去对付 null,让我们看看这些方法。

NPE 是运行期异常,只会在系统运行期间造成,所以导致代码检查无法提前发现它。如果我们能想办法把在运行期出现的 NPE,提前在编译代码时探测到,那么我们就会大大减轻 NPE 对系统造成的损害。


2. 横空出世的注解

@NonNull 这个注解就是一个标记,这个标记可以和 IDE 联动:当可能出现 NPE 时,IDE 会标出警告。

我们先看一段代码:

上面的代码没有加入 @NonNull,可以看到 IDE 并没有给出什么警告。

让我们加上 @NonNull 注解看看:

可以看到,Idea 和 @NonNull 注解形成了联动,并给出了可能出现 NPE 的警告。

有了这个警告,其实对一个复杂的项目来说还不够,因为这些警告很容易就会被忽略过去了,即使忽略了,项目依然可以编译运行起来。

那么,我们是不是可以再增加一步检查?当检查到了可疑的 NPE,根本不允许编译通过。是时候给大家介绍一下 findbugs 了!

3. findbugs 出场了

我们先在 maven 中配置好 findbugs:

<?xml version="1.0" encoding="UTF-8"?>
<project xmlns="http://maven.apache.org/POM/4.0.0"
         xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
         xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
    <modelVersion>4.0.0</modelVersion>

    <groupId>com.github</groupId>
    <artifactId>leetcodeMaster</artifactId>
    <version>1.0-SNAPSHOT</version>
    <build>
        <resources>
            <resource>
                <directory>src/main/resources</directory> <!--扫描resources包下的配置文件-->
                <filtering>true</filtering>
                <includes>
                    <include>**/*.xml</include>
                    <include>**/*.properties</include>
                </includes>
            </resource>
            <resource>
                <directory>src/main/java</directory><!--扫描java包下的配置文件-->
                <includes>
                    <include>**/*.xml</include>
                    <include>**/*.properties</include>
                </includes>
            </resource>
        </resources>
        <plugins>
            <plugin>
                <groupId>org.apache.maven.plugins</groupId>
                <artifactId>maven-compiler-plugin</artifactId>
                <configuration>
                    <source>8</source>
                    <target>8</target>
                </configuration>
            </plugin>
            <plugin>
                <groupId>org.codehaus.mojo</groupId>
                <artifactId>findbugs-maven-plugin</artifactId>
                <version>3.0.5</version>
                <configuration>
                    <!-- 设置分析工作的等级,可以为Min、Default和Max -->
                    <effort>Low</effort>
                    <!-- Low、Medium和High (Low最严格) High只扫描严重错误。建议用Medium-->
                    <threshold>Medium</threshold>
                    <failOnError>true</failOnError>
                    <includeTests>true</includeTests>
                    <!--findbugs需要忽略的错误的配置文件-->
<!--                    <excludeFilterFile>conf/findbugs-exclude-filter.xml</excludeFilterFile>-->
                    <!--findbugs需要忽略的错误的配置文件-->
                    <includeFilterFile>conf/findbugs-include-filter.xml</includeFilterFile>
                </configuration>
                <executions>
                    <execution>
                        <id>run-findbugs</id>
                        <!-- 在package(也可设为compile) 阶段触发执行findbugs检查,比如执行 mvn clean package -->
                        <phase>compile</phase>
                        <goals>
                            <goal>check</goal>
                        </goals>
                    </execution>
                </executions>
            </plugin>
        </plugins>
    </build>
    <dependencies>
        <!-- https://mvnrepository.com/artifact/com.google.guava/guava -->
        <dependency>
            <groupId>com.google.guava</groupId>
            <artifactId>guava</artifactId>
            <version>19.0</version>
        </dependency>
        <!-- https://mvnrepository.com/artifact/org.apache.commons/commons-lang3 -->
        <dependency>
            <groupId>org.apache.commons</groupId>
            <artifactId>commons-lang3</artifactId>
            <version>3.3.2</version>
        </dependency>
        <!-- https://mvnrepository.com/artifact/com.google.code.findbugs/jsr305 -->
        <dependency>
            <groupId>com.google.code.findbugs</groupId>
            <artifactId>jsr305</artifactId>
            <version>3.0.2</version>
        </dependency>
    </dependencies>
</project>

紧接着运行maven,对项目进行编译。
mvn clean compile findbugs:findbugs

可以看到,findbugs 发现可能会在运行期间出现 NPE 后,中断了项目构建过程。

我们再打开 findbugs 的界面看看具体的报错位置:

你瞧,findbugs 准确的找到了可能出现 NPE 的根源。

通过以上这些手段,我们尽可能的将 NPE 提前到编译期发现。

但是啊但是,对一个规模庞大且复杂的项目来说,光使用静态代码检查还是不够的。因为类似 findbugs 这种的静态代码检查工具,不可能对每个 NPE 的检查点都检查到位。并且,探测的问题有时候因为业务原因,也会放松检查要求。

别慌,我们可以让静态代码检查再加上一些别的方法,来联手堵住 NPE 问题,这就是我们下面要说的 Optional。

4. 用 Optional 去除二义性

由于铺天盖地的 null 检查,使得 Java 程序员叫苦不堪。于是官方自 Java8 起,参考了 google 的 guava,引入了 Optional 类型用来避免每次繁琐丑陋的 null 检查。

Optional 本质上就是一个容器,这个容器持有了一个变量类型为 T 的值。所以,Optional 这个容器中的值只会有两种情况,要么为类型 T 的变量值,要么为null。

对于可能出现的为 null 的情况,Optional 本身从创建、检查,到抽取、使用,都提供了对应的方法供使用者调用。并采用了意义很明确的方法去排除了null的二义性。

我们看示例代码:

class Player{
    private int id;
    private String name;

    public int getId() {
        return id;
    }

    public void setId(int id) {
        this.id = id;
    }

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }
}

public class Optional4NPE {
    public static void main(String[] args) {
        Optional<Player> optionalPlayer = Optional.ofNullable(null);
        optionalPlayer.ifPresent(u -> System.out.println(u.getName()));
    }
}

以上代码我们使用了一个 Optional 中的 ofNullable,去创建了一个包含了类型为 Player、值为 null 的 Optional 容器。

运行结果:
'Process finished with exit code 0'

运行后,代码没有任何输出,也没有出现 NPE 异常。没有输出的原因是我们传入了一个 null 值,这个 null 表示值不存在。此时,我们调用 Optional 的 ifPresent 方法做了判断,只有存在值时,才会执行打印输出。

接下来,我们把 null 替换成有意义的值看看。

import java.util.Optional;

class Player{
    private int id;
    private String name;

    public int getId() {
        return id;
    }

    public void setId(int id) {
        this.id = id;
    }

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }
}

public class Optional4NPE {
    public static void main(String[] args) {
        Player player = new Player();
        player.setId(1);
        player.setName("demoUser");
        Optional<Player> optionalPlayer = Optional.ofNullable(player);
        optionalPlayer.ifPresent(u -> System.out.println(u.getName()));
    }
}

输出结果:

demoUser

Process finished with exit code 

可以看到,当传入一个我们创建的 player 时,执行了打印输出方法。

上面我们已经发现,通过 Optional 的 ifPresent 方法,我们明确了 null 的含义,明确认定只要值为 null,就表示不存在。那如果一个变量存在,但是没有值或者没有有意义的值呢?

我们把代码改改:

import java.util.Optional;

class Player{
    private int id;
    private String name;

    public int getId() {
        return id;
    }

    public void setId(int id) {
        this.id = id;
    }

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }
}

public class Optional4NPE {
    public static void main(String[] args) {
        Player player = null;
        Player defaultPlayer = new Player();
        defaultPlayer.setId(1);
        defaultPlayer.setName("————undefinedNAME-----");
        Player player1 = Optional.ofNullable(player).orElse(defaultPlayer);
        System.out.println(player1.getName());
    }
}

运行结果如下:

————undefinedNAME-----

Process finished with exit code 0

这里可以看到,我们使用 orElse 方法,当一个变量值为 null 时,返回一个默认值。通过返回默认值,我们明确了 null 的另外一个含义,对象存在,但是可能没有实际意义。

Optional 的出现,大大改善了我们的 Java 代码质量,减少了 NPE 的可能性,并使得代码的可读性大大增强。

通过使用 Optional,开发人员还能非常自然轻松的使用 Null Object Pattern 模式去处理 Null 问题。Optional 是非常值得在项目中大范围使用的。


相关文章

影集制作软件(精美电子相册怎么一键制作,试试这几个工具!)

电子相册制作软件用过几个,映象中完全免费的好像不多,一般都是好看的模板需要开会员啥的,分享四款用过的音乐相册制作软件,模板风格多,操作也比较简单,有需要的朋友可以自己去试试啦~1、清爽视频编辑清爽视频编辑软件作为一款专业的视频编辑软件,支持视频剪辑、拼接合并、裁剪分割、调速、倒放、AI语音识别、加水

2024-09-19 10:31

羽毛球握拍图解教学(羽毛球基本技术——握拍法)

羽毛球握拍是你打羽毛球的第一步,打过羽毛球的都知道羽毛球拍握法正确与否,对于掌握和提高羽毛球技术水平,有着非常重要的影响。一个正确的握拍方法,能让你顺利打出各种角度和远度的羽毛球落点。羽毛球技术中的握拍方法是多种多样的,但是最基本的握拍法有两种,正手握拍法和反手握拍法。 正手握拍法 虎口对着拍柄窄

2024-09-19 10:16

suv和越野车有什么区别吗(城市SUV能越野吗?)

现在有很多人,买了一辆城市SUV,然后逢人就说自己买了一辆越野车,哪天有空去东山越野去。这些人压根就没弄明白城市SUV与越野车的区别,开着这样的车型去越野,我敢保证他有去无回。那么城市SUV和越野车究竟有什么区别呢?下面我们从车型名称、车身结构、外形尺寸、发动机、变速箱、驱动型式、底盘结构、轮胎等几

2024-09-19 10:00

null网络什么意思(烦人的NULL,你可以走开点了)

1. Null 的问题假设现在有一个需要三个参数的方法。其中第一个参数是必须的,后两个参数是可有可无的。第一种情况,在我们调用这个方法的时候,我们只能传入两个参数,对第三个参数,我们在上下文里是没有的,那么我们调用方法的时候,就需要用一个特殊值去告知这个方法:第三个参数我们拿不到,参数是不存在或者不

2024-09-19 09:46

ps路径怎么转换成选区(Photoshop教你如何将路径与选区进行转换)

路径与选区的转换 路径除了可以直接使用路径工具来创建外,还可以将创建好的选区转换为路径。另外,我们也可以将创建的路径转换为选区。1、将选区转换为路径 在Photoshop中创建好选区之后,在“路径”面板中单击“从选区生成工作路径”按钮。如图所示。 这样就可以将创建的选区转换为路径了。如图所示:2、将

2024-09-19 09:31

臭虫喜欢咬什么样的人(臭虫叮咬人时,会传播很多疾病)

臭虫又叫吸血虫,是因为臭虫从虫卵孵化为幼虫以后,就可以叮咬人吸血了,被臭虫叮咬以后又红又肿,奇痒无比,并且抓挠以后,会留下一些黑斑,影响一些爱美女孩子的形象。臭虫在叮咬人的同时,会传播很多疾病,带来更严重的危害。臭虫怕光怕动静,一般都是夜里关灯,人快睡着的时候才出来叮咬人吸血。等你感觉到了被咬,翻身

2024-09-19 09:15

修真境界等级划分详解(远古凡人修仙等阶划分)

凡人修仙是古老而神秘的一种文化现象,追溯到古代中国的历史,可以发现凡人修仙有着非常完整的体系。在这个体系中,不同的修道者有着不同的等阶,在各自的等阶中可以掌握不同层次的修炼法和技能。下面介绍一下凡人修仙中的几个等阶。第一等阶:凡人凡人可以说是凡人修仙的最初阶段,也是修仙者一开始的阶段。这个阶段的修道

2024-09-19 09:01

十字绣勾边怎么勾教程(手工刺绣技法21种常见刺绣针法)

刺绣在中国至少有二三千年历史,是中国古老的手工艺术。发展悠久的刺绣工艺在技法上也相当繁盛,刺绣的针法变法万千,各具特色,历久不衰。随着社会的演进,世界工艺的融合,衍生出越来越多的刺绣针法。不同的针法可以表现出不同的图案,各种针法的相互融合能让刺绣作品表现更灵活更具特色。下面介绍几种目前常见的刺绣针法

2024-09-19 08:46

方位角计算公式怎样录入到excel

选择好要输入公式的单元格,就可以开始输入公式了。首先需要输入一个等号,告知Excel这是一个公式的开始。然后输入公式中的其他内容:函数、函数的参数、数字和文本、单元格引用、运算符、圆括号等内容。如果是普通公式,则按【Enter】键结束输入,否则按【Ctrl+Shift+Enter】组合键,以数组公

2024-09-19 08:31

银行卡被冻结余额显示0什么情况(银行卡可用余额为0是被冻结了吗?)

当我们在使用银行卡进行交易时,有时会遇到银行卡可用余额为0的情况。很多人会认为这是银行卡被冻结了,但实际上并非总是如此。首先,我们需要了解银行卡的基本工作原理。银行卡是与银行账户相连的电子支付工具,用于方便快捷地进行支付和转账。当我们使用银行卡进行交易时,银行会在我们的账户中扣除相应的资金。因此,银

2024-09-19 08:15