Java中变量的内存地址

问题

请看下面的图片。当我们用java创建一个带有new关键字的对象时,我们从OS获取一个内存地址。

当我们写out.println(objName)时,我们可以看到一个"特殊"字符串作为输出。我的问题是:

  • 这是什么输出?
  • 如果它是操作系统给我们的内存地址:a)如何将此字符串转换为二进制? b)如何获得一个整数变量地址?

alt text


#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(),它将打印完全不同的东西。