Java中不一致的操作会扩大规则

总览

当您在Java中执行一元或二进制操作时,标准行为是使用最宽的操作数(或对byteshortchar使用更宽的操作数)。 这很容易理解,但是如果考虑最佳类型可能会造成混淆。

乘法

当执行乘法运算时,您得到的数值通常比单个数值大得多。 即| a * b | >> | a | 和| a * b | >> | b | 通常是这种情况。 对于小型字体,这可以按预期工作。

考虑以下程序:

public static void main(String[] args) throws IOException {
    System.out.println(is(Byte.MAX_VALUE * Byte.MAX_VALUE));
    System.out.println(is(Short.MAX_VALUE * Short.MAX_VALUE));
    System.out.println(is(Character.MAX_VALUE * Character.MAX_VALUE));
    System.out.println(is(Integer.MAX_VALUE * Integer.MAX_VALUE));
    System.out.println(is(Long.MAX_VALUE * Long.MAX_VALUE));
}

static String is(byte b) {
    return "byte: " + b;
}

static String is(char ch) {
    return "char: " + ch;
}

static String is(short i) {
    return "short: " + i;
}

static String is(int i) {
    return "int: " + i;
}

static String is(long l) {
    return "long: " + l;
}

打印:

int: 16129
int: 1073676289
int: -131071
int: 1
long: 1

只有byte * byteshort * short不会溢出,因为它们已被扩展。 char * char即使允许,也不是有意义的操作。 但是int * int确实会溢出,即使我们有一个长类型也可以存储此值而不会溢出。 byteshort都隐式加宽,但不是int 。 应该真正地将long扩展,但是我们没有一个更宽泛的基本类型,这曾经是有意义的,但是如今64位基本类型看起来并不那么长。

在除数可以扩大结果的意义上,除法有些奇怪。 除数比分子宽并不意味着结果会更大(但通常会更小)

System.out.println(is(Byte.MAX_VALUE / (byte) 1));
System.out.println(is(Byte.MAX_VALUE / (short) 1));
System.out.println(is(Byte.MAX_VALUE / (char) 1));
System.out.println(is(Byte.MAX_VALUE / (int) 1));
System.out.println(is(Byte.MAX_VALUE/ (long) 1));

版画

int: 127
int: 127
int: 127
int: 127
long: 127

当你把一个byte/byte你会得到一个int即使你不能得到比一个大的值byte 。 (除非您将Byte.MIN_VALUE除以-1(在这种情况下, short会这样做),并且如果您将byte/long除以byte/long即使值仍然不能大于byte您也会得到long

模量

当执行模数a % b ,结果不能大于b 。 但是模数会扩大结果而不是减少结果。

System.out.println(is(Byte.MAX_VALUE % Byte.MAX_VALUE));
System.out.println(is(Byte.MAX_VALUE % Short.MAX_VALUE));
System.out.println(is(Byte.MAX_VALUE % Character.MAX_VALUE));
System.out.println(is(Byte.MAX_VALUE % Integer.MAX_VALUE));
System.out.println(is(Byte.MAX_VALUE % Long.MAX_VALUE));
    
System.out.println(is(Byte.MAX_VALUE % (byte) 2));
System.out.println(is(Short.MAX_VALUE % (byte) 2));
System.out.println(is(Character.MAX_VALUE % (byte) 2));
System.out.println(is(Integer.MAX_VALUE % (byte) 2));
System.out.println(is(Long.MAX_VALUE % (byte) 2));

版画

int: 0
int: 127
int: 127
int: 127
long: 127
int: 1
int: 1
int: 1
int: 1
long: 1

如果将X乘以一个数,结果将不会比X宽/大,只能变小。 但是,捷豹路虎say必须扩大范围。 如果将X乘以一个byte ,则结果只能在一个byte的范围内。

我还提到了一元运算,也许最简单的是一元减号。

System.out.println(is(-Byte.MIN_VALUE));
System.out.println(is(-Short.MIN_VALUE));
System.out.println(is(-Character.MIN_VALUE));
System.out.println(is(-Integer.MIN_VALUE));
System.out.println(is(-Long.MIN_VALUE));

版画

int: 128
int: 32768
int: 0
int: -2147483648
long: -9223372036854775808

在前三种情况下,类型扩大了。 一个byte可以扩大到short ,但是作为一个int是正确的。 但是对于intlong ,它并没有扩大,您可能会遇到罕见的溢出。

一元奇数是一元加号,它不会更改值(因此无法更改其范围),但可以扩大值。

System.out.println(is(+Byte.MIN_VALUE));
System.out.println(is(+Short.MIN_VALUE));
System.out.println(is(+Character.MIN_VALUE));
System.out.println(is(+Integer.MIN_VALUE));
System.out.println(is(+Long.MIN_VALUE));

版画

int: -128
int: -32768
int: 0
int: -2147483648
long: -9223372036854775808

我们可以解决这个问题吗?

不幸的是没有。 有太多代码依赖于此逻辑。 例如说你写这样的东西。

long i = ...
byte b = ...
long l = i % b + Integer.MAX_VALUE;

如果i%b要从long变为byte ,则此表达式可能会溢出。

结论

Java可以在需要时扩展某些值,但也不能扩展某些int操作,这实际上应该很long 。 即使这可能更合逻辑,也永远不会给出更窄的结果。

我们需要做的是了解边缘情况,尤其是int * int ,并在看到这样的操作时知道自己扩大它们。 例如

long l = (long) a * b;

除非我们确信a * b将适合int值。

翻译自: https://www.javacodegeeks.com/2015/02/inconsistent-operation-widen-rules-java.html

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值