Integer 与 Long 数字类型的比较:Java与Kotlin的细节不同

简介: 编程语言还是比较傻的。我们在数学中,123 == 123 , 直觉上是一目了然的。但是到了计算机编程语言中, 问题就显得有点“傻瓜”化了。值得一提的下面的表达式:new Long(10).equals(new Integer(10))始终是 false,这确实是一个违背数学常理的“坑”。

编程语言还是比较傻的。

我们在数学中,123 == 123 , 直觉上是一目了然的。但是到了计算机编程语言中, 问题就显得有点“傻瓜”化了。

值得一提的下面的表达式:

new Long(10).equals(new Integer(10))

始终是 false,这确实是一个违背数学常理的“坑”。

再比如,在Java中

    static void test2() {
        // Integer的自动拆装箱的陷阱(整型数-128到127的值比较问题)
        out.println("-------------------");
        Integer x = new Integer(123);
        Long y = new Long(123);
        //out.println(x == y); // Error:(43, 23) java: incomparable types: java.lang.Integer and java.lang.Long
        out.println(x.equals(y)); // false

        out.println("-------------------");
        Integer c = Integer.valueOf(128);
        Long d = Long.valueOf(128);
        //System.out.println(c == d);//Error:(49, 30) java: incomparable types: java.lang.Integer and java.lang.Long
        System.out.println(c.equals(d)); // false
    }

返回的都是false。 因为这个equals方法实现的逻辑还是僵化的计算机编程逻辑。(注意 “数据类型” 这个概念)


    public boolean equals(Object obj) {
        if (obj instanceof Integer) {
            return value == ((Integer)obj).intValue();
        }
        return false;
    }

    public boolean equals(Object obj) {
        if (obj instanceof Long) {
            return value == ((Long)obj).longValue();
        }
        return false;
    }

是不是有点傻?

还好,有个Comparable接口:

public final class Long extends Number implements Comparable<Long> {}
public final class Integer extends Number implements Comparable<Integer> {}

但是,为什么 java.lang.Number 自己不实现 Comparable 呢?如果实现了,我们不就能进行排序 Number 与 Collections.sort,似乎有点奇怪。

此外,与真正基元类型 (float,double) 确定如果两个值相等,也很棘手,要做一个可接受的误差幅度内。请尝试如下代码:

double d1 = 1.0d;
double d2 = 0.0d;
for (int i=0; i<10; i++) {
  d2 += 0.1d;
}
System.out.println(d2 - d1); // -1.1102230246251565E-16, 浮点数会有误差

和你会留下一些小的差别。

所以回到这一问题作出的 Number Comparable。您将如何实施?使用类似 doubleValue() 不可靠。请记住,Number 子类型是:

Byte;
Short;
Integer;
Long;
AtomicInteger;
AtomicLong;
Float;
Double;
BigInteger;和
BigDecimal.

可能您代码可靠 compareTo() 为一系列的如果不下放的方法假如语句吗?Number 实例只能有六种方法向他们提供:

byteValue();
shortValue();
intValue();
longValue();
floatValue();和
doubleValue().

看到了吧,计算机总是那么“傻”,不像人脑那样“智能”。

在Long.java中,提供了一个compareTo方法

    public int compareTo(Long anotherLong) {
        return compare(this.value, anotherLong.value);
    }

但是,仍然局限在Long类型之间比较。就是说,下面的代码 Error:(53, 33) 依然编译不通过

        Integer c = Integer.valueOf(128);
        Long d = Long.valueOf(128);
        //System.out.println(c == d);//Error:(49, 30) java: incomparable types: java.lang.Integer and java.lang.Long
        out.println(d.equals(c));
        out.println(d.compareTo(c)); // Error:(53, 33) java: incompatible types: java.lang.Integer cannot be converted to java.lang.Long
    }

在Kotlin中,Long类型实现了多个compareTo方法,稍微方便了数字之间的比较

    public operator fun compareTo(other: Byte): Int

    public operator fun compareTo(other: Short): Int

    public operator fun compareTo(other: Int): Int

    public override operator fun compareTo(other: Long): Int

    public operator fun compareTo(other: Float): Int

    public operator fun compareTo(other: Double): Int

Kotlin中,Int类型与Long类型之间比较大小:

package com.easy.kotlin

fun main(args: Array<String>) {
    test1()
}

fun test1() {
    val x: Int = 123
    val y: Int = 123

    println(x == y)
    println(x === y)

    val z: Int = 456
    val w: Int = 456
    println(z == w)
    println(z === w)

    val a: Long = 789
    val b: Int = 1010
    println(a<b)
    //println(a!=b) //Error:(22, 13) Kotlin: Operator '!=' cannot be applied to 'Long' and 'Int'
    //println(a==b) //Error:(23, 13) Kotlin: Operator '==' cannot be applied to 'Long' and 'Int'
    println(a.compareTo(b))
}

输出:

true
true
true
true
true
-1

源代码工程:https://github.com/EasyKotlin/kotlin_tutorials

相关文章
|
7月前
|
安全 Java 编译器
Java类型提升与类型转换详解
本文详解Java中的类型提升与类型转换机制,涵盖类型提升规则、自动类型转换(隐式转换)和强制类型转换(显式转换)的使用场景与注意事项。内容包括类型提升在表达式运算中的作用、自动转换的类型兼容性规则,以及强制转换可能引发的数据丢失和运行时错误。同时提供多个代码示例,帮助理解byte、short、char等类型在运算时的自动提升行为,以及浮点数和整型之间的转换技巧。最后总结了类型转换的最佳实践,如避免不必要的转换、使用显式转换提高可读性、金融计算中使用BigDecimal等,帮助开发者写出更安全、高效的Java代码。
369 0
|
7月前
|
安全 IDE Java
Java记录类型(Record):简化数据载体类
Java记录类型(Record):简化数据载体类
526 143
|
7月前
|
Java 测试技术
Java浮点类型详解:使用与区别
Java中的浮点类型主要包括float和double,它们在内存占用、精度范围和使用场景上有显著差异。float占用4字节,提供约6-7位有效数字;double占用8字节,提供约15-16位有效数字。float适合内存敏感或精度要求不高的场景,而double精度更高,是Java默认的浮点类型,推荐在大多数情况下使用。两者都存在精度限制,不能用于需要精确计算的金融领域。比较浮点数时应使用误差范围或BigDecimal类。科学计算和工程计算通常使用double,而金融计算应使用BigDecimal。
2628 102
|
8月前
|
存储 缓存 人工智能
Java int和Integer的区别
本文介绍了Java中int与Integer的区别及==与equals的比较机制。Integer是int的包装类,支持null值。使用==比较时,int直接比较数值,而Integer比较对象地址;在-128至127范围内的Integer值可缓存,超出该范围或使用new创建时则返回不同对象。equals方法则始终比较实际数值。
270 0
|
5月前
|
存储 算法 安全
Java集合框架:理解类型多样性与限制
总之,在 Java 题材中正确地应对多样化与约束条件要求开发人员深入理解面向对象原则、范式编程思想以及JVM工作机理等核心知识点。通过精心设计与周密规划能够有效地利用 Java 高级特征打造出既健壮又灵活易维护系统软件产品。
158 7
|
6月前
|
Java 开发者
Java 函数式编程全解析:静态方法引用、实例方法引用、特定类型方法引用与构造器引用实战教程
本文介绍Java 8函数式编程中的四种方法引用:静态、实例、特定类型及构造器引用,通过简洁示例演示其用法,帮助开发者提升代码可读性与简洁性。
|
7月前
|
安全 算法 Java
Java泛型编程:类型安全与擦除机制
Java泛型详解:从基础语法到类型擦除机制,深入解析通配符与PECS原则,探讨运行时类型获取技巧及最佳实践,助你掌握泛型精髓,写出更安全、灵活的代码。
|
11月前
|
IDE Java 开发工具
JetBrains IntelliJ IDEA 2025.1 发布 - 领先的 Java 和 Kotlin IDE
JetBrains IntelliJ IDEA 2025.1 (macOS, Linux, Windows) - 领先的 Java 和 Kotlin IDE
683 2
|
12月前
|
存储 传感器 缓存
java变量与数据类型:整型、浮点型与字符类型
### Java数据类型全景表简介 本文详细介绍了Java的基本数据类型和引用数据类型,涵盖每种类型的存储空间、默认值、取值范围及使用场景。特别强调了`byte`、`int`、`long`、`float`、`double`等基本类型在不同应用场景中的选择与优化,如文件流处理、金融计算等。引用数据类型部分则解析了`String`、数组、类对象、接口和枚举的内存分配机制。
429 15
|
12月前
|
Java
课时11:Java数据类型划分(浮点类型)
课时11介绍了Java中的浮点数据类型。主要内容包括:1. 定义小数,默认使用Double类型;2. 定义Float变量,需在数值后加&quot;F&quot;或&quot;f&quot;进行强制转换;3. 观察不同类型计算结果,如Int型除法会丢失精度,需至少包含一个Double或Float类型以确保准确性。总结指出,在复杂计算中推荐使用Double类型以避免精度损失。
281 5