当前位置: 首页 > 面试题库 >

Java条件运算符?:结果类型

江丰羽
2023-03-14
问题内容

我对条件运算符有些疑惑。考虑以下两行:

Float f1 = false? 1.0f: null;
Float f2 = false? 1.0f: false? 1.0f: null;

为什么f1变为null且第二条语句引发NullPointerException?

Langspec-3.0第15.25段说:

否则,第二和第三操作数分别为S1和S2类型。令T1为对S1进行装箱转换所产生的类型,而T2为对S2进行装箱转换所产生的类型。条件表达式的类型是将捕获转换(§5.1.10)应用于lub(T1,T2)(§15.12.2.7)的结果。

因此,对于false?1.0f:nullT1是Float,而T2是null类型。但是,结果是lub(T1,T2)什么?第15.12.2.7段的内容有点太多…

顺便说一句,我在Windows上使用1.6.0_18。

PS:我知道那Float f2 = false? (Float) 1.0f: false? (Float) 1.0f: null;不会抛出NPE。


问题答案:

区别在于在编译时静态键入表达式:

摘要

E1: `(false ? 1.0f : null)`
    - arg 2 '1.0f'           : type float,
    - arg 3 'null'           : type null 
    - therefore operator ?:  : type Float (see explanation below)
    - therefore autobox arg2
    - therefore autobox arg3

E2: `(false ? 1.0f : (false ? 1.0f : null))`
    - arg 2 '1.0f'                    : type float
    - arg 3 '(false ? 1.0f : null)'   : type Float (this expr is same as E1)
    - therefore, outer operator ?:    : type float (see explanation below)
    - therefore un-autobox arg3

详细说明:

通过阅读规格书以及从您获得的结果开始倒推,这是我的理解。它归结为f2的第三操作数的类型
的条件是空型,而f2的第三操作数的类型 的条件被认为是浮点。

注意: 重要的是要记住,类型的确定和装箱/拆箱代码的插入是在编译时完成的。装箱/拆箱代码的实际执行在运行时完成。

Float f1 = (false ? 1.0f : null);
Float f2 = (false ? 1.0f : (false ? 1.0f : null));

f1条件和f2内部条件: (false?1.0f:null)

f1条件和f2内部条件相同: (false?1.0f:null) 。f1条件和f2内部条件中的操作数类型为:

type of second operand = float
type of third operand = null type (§4.1)

§15.25中的大多数规则都已通过,并且最终评估确实适用:

否则,第二和第三操作数分别为S1和S2类型。令T1为对S1进行装箱转换所产生的类型,而T2为对S2进行装箱转换所产生的类型。条件表达式的类型是将捕获转换(第5.1.10节)应用于lub(T1,T2)(第15.12.2.7节)的结果。

S1 = float
S2 = null type
T1 = Float
T2 = null type
type of the f1 and f2 inner conditional expressions = Float

因为对于f1,赋值是给Float参考变量的,所以表达式(空)的结果已成功赋值。

对于f2外部条件:( 假?1.0f:[f2内部条件])

对于f2外部条件,类型为:

type of second operand = float
type of third operand = Float

请注意,与直接引用 文字的f1 /
f2内部条件相比,操作数类型有所不同(第4.1节)。由于具有2个数字可转换类型的区别,因此适用第15.12.2.7节的规则:

* 否则,如果第二个和第三个操作数具有可转换为数字类型(第5.1.8节)的类型,则有以下几种情况:…

*

否则,将二进制数值提升(第5.6.2节)应用于操作数类型,条件表达式的类型是第二和第三操作数的提升类型。请注意,二进制数值升级执行
拆箱转换
(第5.1.8节)和值集转换(第5.1.13节)。

由于对f2内部条件(空)的结果执行了拆箱转换,因此引发了NullPointerException。



 类似资料:
  • 我对条件运算符有点困惑。请考虑以下两行: 为什么f1变成null,而第二条语句抛出NullPointerException? LANSPEC-3.0第15.25段SAIS: 否则,第二和第三操作数的类型分别为S1和S2。设T1是对S1应用装箱转换得到的类型,设T2是对S2应用装箱转换得到的类型。条件表达式的类型是将捕获转换(§5.1.10)应用于lub(T1,T2)(§15.12.2.7)的结果。

  • Java 提供了一个特别的三元运算符(也叫三目运算符)经常用于取代某个类型的 if-then-else 语句。 条件运算符的符号表示为“?:”,使用该运算符时需要有三个操作数,因此称其为三目运算符。使用条件运算符的一般语法结构为: 其中,expression 是一个布尔表达式。当 expression 为真时,执行 statement1, 否则就执行 statement3。此三元运算符要求返回一个

  • 主要内容:RxJava 条件运算符 介绍,RxJava 条件运算符 示例RxJava 条件运算符 介绍 以下是用于 Observable 的条件运算符。 运算符 描述 All 评估发出的所有项目以满足给定标准。 Amb 仅在给定多个 Observable 的情况下从第一个 Observable 发出所有项目。 Contains 检查 Observable 是否发出特定项目。 DefaultIfEmpty 如果 Observable 不发出任何内容,则发出默认项。 Se

  • GCC(4.8.2版)手册中规定了以下内容: < code >-ftree-loop-if-convert-stores :< br >尝试同时if-convert包含内存写入的条件跳转。这种转换对于多线程程序来说是不安全的,因为它将有条件的内存写入转换为无条件的内存写入。举个例子, 被转换为 可能产生数据竞争。 然而,我想知道使用与语句。 < li >在第一段代码中,只有满足条件时,< code

  • 条件运算符的格式为:A ? B : C。其中A是布尔类型,如果表达式A的执行结果为true,则执行B,并返回表达式B的执行结果,如果A的执行结果为false,则执行C,并返回表达式C的执行结果。例如${("a" > "b")? 4: 5}的执行结果为5。

  • C++ 运算符 Exp1 ? Exp2 : Exp3; 其中,Exp1、Exp2 和 Exp3 是表达式。请注意冒号的使用和位置。? : 表达式的值取决于 Exp1 的计算结果。如果 Exp1 为真,则计算 Exp2 的值,且 Exp2 的计算结果则为整个 ? : 表达式的值。如果 Exp1 为假,则计算 Exp3 的值,且 Exp3 的计算结果则为整个 ? : 表达式的值。 ? 被称为三元运