草庐IT

java - Java如何处理原始数字和引用数字之间的长比较?

coder 2024-04-02 原文

我已经在笔记本电脑(Oracle HotSpot JVM、JDK 1.8、64 位)上尝试了以下源代码:

Long l;
Long l1 = 100L;
Long l2 = 100L;
System.out.println(Long.valueOf(100L) == Long.valueOf(100L));
System.out.println((l = 100L) == Long.valueOf(100L));
System.out.println(l1 == l2);
System.out.println(Long.valueOf(128L) == Long.valueOf(128L));
System.out.println(Long.valueOf(129L) == 129L);
System.out.println(Long.valueOf(255L) == new Long(255L));

然后我在IntelliJ IDEA Community 2019.1中反编译源码得到如下内容:

Long l1 = 100L;
Long l2 = 100L;
System.out.println(100L == 100L);
System.out.println(100L == 100L);
System.out.println(l1 == l2);
System.out.println(128L == 128L);
System.out.println(Long.valueOf(129L) == 129L);
System.out.println(255L == new Long(255L));

我得到了答案:

true
true
true
false
true
false

我已经知道从原始数到对应的引用数的整数赋值会自动装箱。并且如果数字在[-128,127],就会使用缓存,这意味着第4、5、6、7、9行的结果是合理的。

不过,我很好奇Java是如何处理第8行的引用数和原始数的比较的?换句话说,原始数字实际上是如何存储在 Oracle HotSpot JVM 中的内存中的?

到目前为止我还没有找到任何帮助。任何建议将不胜感激。


更新于 2019-04-02 19:25:24: 我试图显示 Long.valueOf(129L)129L 的地址,如下所示:

System.out.println(System.identityHashCode(Long.valueOf(129L)));
System.out.println(System.identityHashCode(129L));
System.out.println(System.identityHashCode(129L));

我得到以下信息:

731260860
1709366259
1335298403

它们显然是不同的对象,即使看起来它们是相同的原始数,129L

最佳答案

关于 .class 文件反编译的任何奇怪的事情都可能无关紧要。它们可能只是表示您使用的反编译器不可靠。所以让我们忽略你的反编译代码。


测试基本类型和盒装数字类型是否相等的一般规则如下(简单版):

  • <primitive-number> == <primitive-number> :

    1. 如果值不是同一类型,则将“较小”的类型转换为较大的类型。
    2. 使用原始类型的排序进行比较
  • <primitive-number> == <boxed-number><boxed-number> == <primitive-number> :

    1. 将装箱的数字拆箱到它对应的原始类型
    2. 如果值不是同一类型,则将“较小”的类型转换为较大的类型。
    3. 使用原始类型的排序进行比较
  • <boxed-number> == <boxed-number>

    1. 如果盒装类型不同:编译错误
    2. 如果装箱的类型相同,则根据引用相等进行比较。

JLS 11 的相关部分是 15.21.1对于前两个项目符号和 15.21.3对于第三个。 (您会发现从 Java 5 开始的所有版本的 JLS 基本上都说同样的话。)

复杂的问题是两次装箱相同的原始值可能会也可能不会给你相同的引用。它取决于 JVM 命令行开关 (!) 的类型和值以及(在某些情况下)。

类型 Byte , Short , IntegerLong都为它们的值空间的子范围维护一个装箱值的缓存。通常,子范围是从 -128 到 +127(含),但这未指定。如果您自动装箱该范围内的原语,您将获得缓存的值。如果您使用 <Type>.valueOf(<prim-type>),这同样适用转换该范围内的值1。但是如果你调用new <Type>(<prim-type>)你将*总是得到一个新创建的对象。

要注意的是,使用 == 不是一个好主意。比较两个盒装数字,因为您得到的答案通常是不可预测的。


1 - 这是因为装箱指定使用 valueOf方法。

关于java - Java如何处理原始数字和引用数字之间的长比较?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/55473101/

有关java - Java如何处理原始数字和引用数字之间的长比较?的更多相关文章

  1. ruby-on-rails - Rails 应用程序之间的通信 - 2

    我构建了两个需要相互通信和发送文件的Rails应用程序。例如,一个Rails应用程序会发送请求以查看其他应用程序数据库中的表。然后另一个应用程序将呈现该表的json并将其发回。我还希望一个应用程序将存储在其公共(public)目录中的文本文件发送到另一个应用程序的公共(public)目录。我从来没有做过这样的事情,所以我什至不知道从哪里开始。任何帮助,将不胜感激。谢谢! 最佳答案 无论Rails是什么,几乎所有Web应用程序都有您的要求,大多数现代Web应用程序都需要相互通信。但是有一个小小的理解需要你坚持下去,网站不应直接访问彼此

  2. ruby - Ruby 的 Hash 在比较键时使用哪种相等性测试? - 2

    我有一个围绕一些对象的包装类,我想将这些对象用作散列中的键。包装对象和解包装对象应映射到相同的键。一个简单的例子是这样的:classAattr_reader:xdefinitialize(inner)@inner=innerenddefx;@inner.x;enddef==(other)@inner.x==other.xendenda=A.new(o)#oisjustanyobjectthatallowso.xb=A.new(o)h={a=>5}ph[a]#5ph[b]#nil,shouldbe5ph[o]#nil,shouldbe5我试过==、===、eq?并散列所有无济于事。

  3. java - 等价于 Java 中的 Ruby Hash - 2

    我真的很习惯使用Ruby编写以下代码:my_hash={}my_hash['test']=1Java中对应的数据结构是什么? 最佳答案 HashMapmap=newHashMap();map.put("test",1);我假设? 关于java-等价于Java中的RubyHash,我们在StackOverflow上找到一个类似的问题: https://stackoverflow.com/questions/22737685/

  4. ruby-on-rails - Enumerator.new 如何处理已通过的 block ? - 2

    我在理解Enumerator.new方法的工作原理时遇到了一些困难。假设文档中的示例:fib=Enumerator.newdo|y|a=b=1loopdoy[1,1,2,3,5,8,13,21,34,55]循环中断条件在哪里,它如何知道循环应该迭代多少次(因为它没有任何明确的中断条件并且看起来像无限循环)? 最佳答案 Enumerator使用Fibers在内部。您的示例等效于:require'fiber'fiber=Fiber.newdoa=b=1loopdoFiber.yieldaa,b=b,a+bendend10.times.m

  5. ruby - #之间? Cooper 的 *Beginning Ruby* 中的错误或异常 - 2

    在Cooper的书BeginningRuby中,第166页有一个我无法重现的示例。classSongincludeComparableattr_accessor:lengthdef(other)@lengthother.lengthenddefinitialize(song_name,length)@song_name=song_name@length=lengthendenda=Song.new('Rockaroundtheclock',143)b=Song.new('BohemianRhapsody',544)c=Song.new('MinuteWaltz',60)a.betwee

  6. ruby-on-rails - `a ||= b` 和 `a = b if a.nil 之间的区别? - 2

    我正在检查一个Rails项目。在ERubyHTML模板页面上,我看到了这样几行:我不明白为什么不这样写:在这种情况下,||=和ifnil?有什么区别? 最佳答案 在这种特殊情况下没有区别,但可能是出于习惯。每当我看到nil?被使用时,它几乎总是使用不当。在Ruby中,很少有东西在逻辑上是假的,只有文字false和nil是。这意味着像if(!x.nil?)这样的代码几乎总是更好地表示为if(x)除非期望x可能是文字false。我会将其切换为||=false,因为它具有相同的结果,但这在很大程度上取决于偏好。唯一的缺点是赋值会在每次运行

  7. ruby - 查找字符串中的内容类型(数字、日期、时间、字符串等) - 2

    我正在尝试解析一个CSV文件并使用SQL命令自动为其创建一个表。CSV中的第一行给出了列标题。但我需要推断每个列的类型。Ruby中是否有任何函数可以找到每个字段中内容的类型。例如,CSV行:"12012","Test","1233.22","12:21:22","10/10/2009"应该产生像这样的类型['integer','string','float','time','date']谢谢! 最佳答案 require'time'defto_something(str)if(num=Integer(str)rescueFloat(s

  8. ruby - 一个 YAML 对象可以引用另一个吗? - 2

    我想让一个yaml对象引用另一个,如下所示:intro:"Hello,dearuser."registration:$introThanksforregistering!new_message:$introYouhaveanewmessage!上面的语法只是它如何工作的一个例子(这也是它在thiscpanmodule中的工作方式。)我正在使用标准的ruby​​yaml解析器。这可能吗? 最佳答案 一些yaml对象确实引用了其他对象:irb>require'yaml'#=>trueirb>str="hello"#=>"hello"ir

  9. java - 从 JRuby 调用 Java 类的问题 - 2

    我正在尝试使用boilerpipe来自JRuby。我看过guide从JRuby调用Java,并成功地将它与另一个Java包一起使用,但无法弄清楚为什么同样的东西不能用于boilerpipe。我正在尝试基本上从JRuby中执行与此Java等效的操作:URLurl=newURL("http://www.example.com/some-location/index.html");Stringtext=ArticleExtractor.INSTANCE.getText(url);在JRuby中试过这个:require'java'url=java.net.URL.new("http://www

  10. java - 我的模型类或其他类中应该有逻辑吗 - 2

    我只想对我一直在思考的这个问题有其他意见,例如我有classuser_controller和classuserclassUserattr_accessor:name,:usernameendclassUserController//dosomethingaboutanythingaboutusersend问题是我的User类中是否应该有逻辑user=User.newuser.do_something(user1)oritshouldbeuser_controller=UserController.newuser_controller.do_something(user1,user2)我

随机推荐