java怎么输入一个值
【CSDN 编者按】Kotlin 和 Java 是如何解决 Null 问题?本文作者分享了解决思路。
原文链接:https://blog.frankel.ch/null-safety-java-vs-kotlin/
未经授权,禁止转载!
作者 | Nicolas Fränkel 责编 | 弯月出品 | CSDN(ID:CSDNnews)在本文中,我想讨论一下 Kotlin 和 Java 是如何解决 Null 问题的。
可为 Null
相信每一位从事软件开发超过两年的人都听过下面这句话:
我把 Null 引用称为自己的十亿美元错误。它的发明是在 1965 年,那时我用一个面向对象语言(ALGOL W)设计了第一个全面的引用类型系统。我的目的是确保所有引用的使用都是绝对安全的,编译器会自动进行检查。但是我未能抵御住诱惑,加入了 Null 引用,仅仅是因为实现起来非常容易。结果导致了数不清的错误、漏洞和系统崩溃,可能在之后的 40 年中造成了十亿美元的损失。
—— 图灵奖得主 Tony Hoare
Null 背后的基本思想是能够定义一个未初始化的变量。当有人调用这类变量的某个成员时,运行时就会寻找变量的内存地址,结果就是引用失败,因为其后面没有任何东西。
许多编程语言都包含 Null 值,只不过名称不同罢了:
Python 有 None;
JavaScript 有 null;
Java、Scala 和 Kotlin 也有 null;
Ruby 有 nil;
以及其他等等。
有些不允许使用未初始化的值,比如 Rust。
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()Java 中的 Null 安全性如上,我们讨论了 Kotlin 管理 Null 值的方法,下面我们来看看 Java。
首先,Java 中既没有不可为 null 的类型,也没有 null 安全的运算符。因此,每个变量都有可能为 null,而且我们也的确应该如此思考。
MyString str = getMyString(); #1 Integer anInt = null; #2if (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。
如果我们将上述方法的返回类型改为 Optional,就可以编写出 null 安全的代码,而且还可以获得不可变性:
final var baz = getFoo().flatMap(Foo::getBar) .flatMap(Bar::getBaz) .orElse(null);对于这种方法,我认为核心问题在于,Optional 本身可以为 null。Java 语言本身无法确保 Optional 不为 null。此外,方法的输入参数不建议使用 Optional。
为了解决这个问题,网上涌现了很多基于注解的库:
然而,不同的库,处理方式也不同:
Spring 会在编译时生成警告消息;
FindBugs 需要专门执行;
Lombok 会生成一段检查 null 的代码,如果变量无论如何都会为 null,则抛出异常 NullPointerException。
总结当 Null 安全性不是一个大问题时,Java 可以被接受。因此,NullPointerException 异常会频繁发生。唯一安全的解决方案是将每个方法调用包装在 null 检查中。这种方式很有效,但同时也很模式化,代码也更加难以阅读。开发人员称赞 Kotlin 带来了 Null 安全性,这是因为该语言的设计中融入了 Null 值处理机制。Java 这方面的处理远不如 Kotlin,因为 Java 语言架构师更加重视向后兼容性,而不是代码安全,这是设计上的决定。但是,作为一名开发人员,从 Null 安全性的角度出发,我认为 Kotlin 是比 Java 更有吸引力的选择。
娄标真576java中如何输入一个数字,这个数字最大为1,000,000,000.请指教 -
凤维容17067206491 ______ 输入整数用long就行,java的long是64位的,能表示正负20亿之内的数值 若是有小数,建议使用BigDecimal
娄标真576如何在java里给变量从键盘输入值? -
凤维容17067206491 ______ 如果你还没有学到异常的部分,那么先不要管这个问题,学到那了就好办了 import java.io.*; class test//主类 { public static void main(String []args)//测试函数 { int a; a=MyInput.readInt(); System.out.print(a); //System.out.println(a+","+b+","+c...
娄标真576java怎么自己输入参数值? -
凤维容17067206491 ______ public static void main (String []args){ double a,b,c; System.out.print("请输入三个数:"); Scanner sc = new Scanner(System.in); a = sc.nextDouble(); b = sc.nextDo...
娄标真576java输入不确定个数的值时应怎样写输入语句 -
凤维容17067206491 ______ import java.io.*; public class standardin1 { public static void main(string[] args) throws ioexception{ system.out.println(" 输入一个字符"); //system.in.read(); //利用read()语句暂缓程序运行 只过滤一个字符 char cc; cc=(char)system.in.read...
娄标真576java输入一串数字求出其中的最大值和最小值 -
凤维容17067206491 ______ 你定义了max=0; 然后输入-1和-2,他们比0大吗?, 你输入的值只有比0大,才会把值赋给max
娄标真576java 如何往一个多组数据里每组数据加一个值 -
凤维容17067206491 ______ 数组大小是固定的,如果加一个值会超出它的容量,那就得新建一个,再把旧的和新增的一个值一起填入
娄标真576java如何获取对话框的值 -
凤维容17067206491 ______ 方法1、在setBombFrame 中添加两个方法,其返回值分别是行值和列值.然后在你说的另一个java文件中使用setBombFrame 的对象实例来调用这两个方法.public String getRow() { return this.s
娄标真576java在命令行中实现输入 -
凤维容17067206491 ______ 这样:import java.io.BufferedReader; import java.io.IOException; import java.io.InputStreamReader; public static void main(String[] args) { InputStreamReader isr = new InputStreamReader(System.in); BufferedReader br = new BufferedReader(isr); ...
娄标真576java中怎么实现可以自己输入一个未知数,然后对其判断该数是否能同时被3、5、7整除 -
凤维容17067206491 ______ import java.util.*;public class Test { public static void main(String[] args) { int a; Scanner input=new Scanner(System.i
娄标真576编写一个java程序,输入一组整数,以 - 1结束,比较并输出其中的最大值和最小值 -
凤维容17067206491 ______ package collection;import java.util.ArrayList;import java.util.Arrays;import java.util.Iterator;import java.util.List;import ja...