Kotlin 与 Java 如何解决 Null 问题?

680302648f004e3604363bffea4b04ea.gif

【CSDN 编者按】Kotlin 和 Java 是如何解决 Null 问题?本文作者分享了解决思路。

原文链接:https://blog.frankel.ch/null-safety-java-vs-kotlin/

未经授权,禁止转载!

作者 | Nicolas Fränkel   责编 | 弯月

出品 | CSDN(ID:CSDNnews)

在本文中,我想讨论一下 Kotlin 和 Java 是如何解决 Null 问题的。

c0c9cf44dddc50e18005ea9689b17816.png

可为 Null

相信每一位从事软件开发超过两年的人都听过下面这句话:

我把 Null 引用称为自己的十亿美元错误。它的发明是在 1965 年,那时我用一个面向对象语言(ALGOL W)设计了第一个全面的引用类型系统。我的目的是确保所有引用的使用都是绝对安全的,编译器会自动进行检查。但是我未能抵御住诱惑,加入了 Null 引用,仅仅是因为实现起来非常容易。结果导致了数不清的错误、漏洞和系统崩溃,可能在之后的 40 年中造成了十亿美元的损失。

                          —— 图灵奖得主 Tony Hoare

Null 背后的基本思想是能够定义一个未初始化的变量。当有人调用这类变量的某个成员时,运行时就会寻找变量的内存地址,结果就是引用失败,因为其后面没有任何东西。

许多编程语言都包含 Null 值,只不过名称不同罢了:

  • Python 有 None;

  • JavaScript 有 null;

  • Java、Scala 和 Kotlin 也有 null;

  • Ruby 有 nil;

  • 以及其他等等。

有些不允许使用未初始化的值,比如 Rust。

cb03d6fe67c8ea948f1b90a189c032ab.png

Kotlin 中的 Null 安全性

如上所示,Kotlin 也有 null 值。只不过,null 融入到了类型系统中。在 Kotlin 中,每个类型 X 实际上都有两种类型:

X:不可为 null,类型 X 的任何变量都不可以为 null。编译器会确保这一点;

val str: String = null

上述代码无法通过编译。

X?:可以为 null。

val str: String? = null

上述代码可以编译。

既然 Kotlin 允许使用 null 值,为什么支持者们会鼓吹它具有 null 安全性呢?因为编译器会调用可能为 null 值(即可为空类型)的成员。

val str: String? = getNullableString()
val int: Int? = str.toIntOrNull()      #1

#1 无法通过编译。

修复上述代码的方式是,在调用成员之前,先检查变量是否为 null:

val str: String? = getNullableString()
val int: Int? = if (str == null) null
          else str.toIntOrNull()

这种方法很模式化,Kotlin 提供了 null 安全的运算符:

val str: String? = getNullableString()
val int: Int? = str?.toIntOrNull()

5d97cf8a1135dc60d1ba23968b043e5c.png

Java 中的 Null 安全性

如上,我们讨论了 Kotlin 管理 Null 值的方法,下面我们来看看 Java。

首先,Java 中既没有不可为 null 的类型,也没有 null 安全的运算符。因此,每个变量都有可能为 null,而且我们也的确应该如此思考。

MyString str = getMyString();         #1  
Integer anInt = null;                 #2
if (str != null) {
    anInt = str.toIntOrNull();
}

#1 String 没有 toIntOrNull() 方法,所以我们假设 MyString 是一个包装类型,实际的操作交给 String。

#2 这里必须使用可变引用。

如果将多个调用放在一起,结果更糟,因为每个返回值都有可能为 null。为了安全着想,我们需要检查每个方法调用返回的值是否为 null。如下代码片段有可能抛出异常 NullPointerException:

var baz = getFoo().getBar().getBaz();

修复方法如下,但非常繁琐:

var foo = getFoo();
var bar = null;
var baz = null;
if (foo != null) {
    bar = foo.getBar();
    if (bar != null) {
        baz = bar.getBaz();
    }
}

出于这个原因,Java 8 引入了 Optional 类型。Optional 是一个包装,负责处理可能为 null 值的情况。在其他语言中,该类型被称为 Maybe、Option 等。

Java 语言的设计者建议,方法应返回:

  • 如果 X 不可能为 null,则返回类型 X;

  • 如果 X 可能为 null,则返回类型 Optional<X>。

如果我们将上述方法的返回类型改为 Optional,就可以编写出 null 安全的代码,而且还可以获得不可变性:

final var baz = getFoo().flatMap(Foo::getBar)
                        .flatMap(Bar::getBaz)
                        .orElse(null);

对于这种方法,我认为核心问题在于,Optional 本身可以为 null。Java 语言本身无法确保 Optional 不为 null。此外,方法的输入参数不建议使用 Optional。

为了解决这个问题,网上涌现了很多基于注解的库:

e5e534931d5c17416f89f459f5f28fa0.png

然而,不同的库,处理方式也不同:

  • Spring 会在编译时生成警告消息;

  • FindBugs 需要专门执行;

  • Lombok 会生成一段检查 null 的代码,如果变量无论如何都会为 null,则抛出异常 NullPointerException。

6a7160be17829b8b23935edc3f926798.png

总结

当 Null 安全性不是一个大问题时,Java 可以被接受。因此,NullPointerException 异常会频繁发生。唯一安全的解决方案是将每个方法调用包装在 null 检查中。这种方式很有效,但同时也很模式化,代码也更加难以阅读。

开发人员称赞 Kotlin 带来了 Null 安全性,这是因为该语言的设计中融入了 Null 值处理机制。Java 这方面的处理远不如 Kotlin,因为 Java 语言架构师更加重视向后兼容性,而不是代码安全,这是设计上的决定。但是,作为一名开发人员,从 Null 安全性的角度出发,我认为 Kotlin 是比 Java 更有吸引力的选择。

 
 

61c62b4c10fead68c14da85fb81e9232.gif

 
 

1be3d8392333c791a4550391afe6ba98.jpeg

☞复旦“首发”国产版 ChatGPT!名为 MOSS 惹群嘲,回应称:还非常不成熟
☞从 ClickHouse 到 Apache Doris,腾讯音乐内容库数据平台架构演进实践
☞大学生快上车!大学教授要求用 ChatGPT 写论文
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
KotlinJava 都是非常流行的编程语言,各自有着一些优势。以下是 KotlinJava 的一些优点: Kotlin 的优势: 1. 更简洁:Kotlin 的语法相对于 Java 更简洁,可以减少代码的冗余和重复性。它引入了许多现代化的语言特性,如类型推断、空安全和扩展函数等,使得代码编写更加简洁高效。 2. 互操作性:Kotlin 可以与 Java 无缝互操作,这意味着开发者可以在现有的 Java 项目中使用 Kotlin,或者在 Kotlin 项目中使用 Java 库。这种互操作性使得迁移到 Kotlin 更加容易,并且可以逐步采用 Kotlin 而不会对现有的代码产生太大影响。 3. 安全性:Kotlin语言级别支持空安全,这意味着开发者可以明确指定一个变量是否可以为 null,从而减少空指针异常的发生。这种特性可以在编译时检测潜在的空指针错误,提高代码的健壮性。 4. 函数式编程支持:Kotlin 支持函数作为一等公民,并且提供了许多函数式编程的特性,如 Lambda 表达式和高阶函数。这使得在 Kotlin 中编写函数式风格的代码更加方便和简洁。 Java 的优势: 1. 广泛应用:Java 是一种非常成熟和广泛应用的编程语言,特别在企业级应用和大规模系统开发方面表现出色。它有一个庞大的开发者社区和丰富的第三方库支持,可以满足各种开发需求。 2. 跨平台性:Java 是一种跨平台的语言,通过 Java 虚拟机 (JVM) 的存在,Java 可以在不同的操作系统上运行。这使得开发者可以编写一次代码,然后在多个平台上运行,提高了开发效率和代码的可移植性。 3. 强类型语言Java 是一种静态强类型语言,它在编译时进行类型检查,可以提供更好的代码安全性和可维护性。类型检查可以帮助开发者在编译阶段捕获潜在的类型错误,减少运行时错误的发生。 4. 丰富的工具生态系统:Java 有着丰富的工具和框架生态系统,如 Spring、Hibernate 等,可以帮助开发者更快速地构建复杂的应用程序。这些工具和框架提供了许多现成的解决方案和最佳实践,使得开发过程更加高效和便捷。 总的来说,KotlinJava 都有各自的优势,并且可以根据具体的需求和项目来选择使用哪种语言

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

CSDN资讯

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值