问题
请看下面的图片。当我们用java创建一个带有new关键字的对象时,我们从OS获取一个内存地址。
当我们写out.println(objName)
时,我们可以看到一个"特殊"字符串作为输出。我的问题是:
- 这是什么输出?
- 如果它是操作系统给我们的内存地址:a)如何将此字符串转换为二进制? b)如何获得一个整数变量地址?
#1 热门回答(130 赞)
这是由'@'字符分隔的类名和System.identityHashCode()。标识哈希码表示的是特定于实现的。它通常是对象的初始内存地址,但随着时间的推移,VM可以在内存中移动对象。所以(简单地说)你不能依赖它作为任何东西。
获取变量的内存地址在Java中是没有意义的,因为JVM可以自由地实现对象并在它看起来合适时移动它们(你的对象可能/将在垃圾收集等过程中移动)
Integer.toBinaryString()将为你提供二进制形式的整数。
#2 热门回答(21 赞)
有可能使用sun.misc.Unsafe
:从@Peter Lawrey看到这个很棒的答案 - > Is there a way to get a reference address?
使用printAddresses()的代码:
public static void printAddresses(String label, Object... objects) {
System.out.print(label + ": 0x");
long last = 0;
int offset = unsafe.arrayBaseOffset(objects.getClass());
int scale = unsafe.arrayIndexScale(objects.getClass());
switch (scale) {
case 4:
long factor = is64bit ? 8 : 1;
final long i1 = (unsafe.getInt(objects, offset) & 0xFFFFFFFFL) * factor;
System.out.print(Long.toHexString(i1));
last = i1;
for (int i = 1; i < objects.length; i++) {
final long i2 = (unsafe.getInt(objects, offset + i * 4) & 0xFFFFFFFFL) * factor;
if (i2 > last)
System.out.print(", +" + Long.toHexString(i2 - last));
else
System.out.print(", -" + Long.toHexString( last - i2));
last = i2;
}
break;
case 8:
throw new AssertionError("Not supported");
}
System.out.println();
}
我设置了这个测试:
//hashcode
System.out.println("Hashcode : "+myObject.hashCode());
System.out.println("Hashcode : "+System.identityHashCode(myObject));
System.out.println("Hashcode (HEX) : "+Integer.toHexString(myObject.hashCode()));
//toString
System.out.println("toString : "+String.valueOf(myObject));
printAddresses("Address", myObject);
这是输出:
Hashcode : 125665513
Hashcode : 125665513
Hashcode (HEX) : 77d80e9
toString : java.lang.Object@77d80e9
Address: 0x7aae62270
结论:
- hashcode!=地址
- toString = class @ HEX(hashcode)
#3 热门回答(11 赞)
这是Object的"toString()"实现的输出。如果你的类重写toString(),它将打印完全不同的东西。