GC复制存活的对象,内存地址会变吗?以前的引用怎么办?

问题

先执行gc方法,10S后打印数组的地址,发现没变,是不是说明复制算法不改变对象的内存地址

解读

toString打印的值 现有的匿名回答是正解:题主做的实验根本没有涉及对象地址。java.lang.Object默认的toString()实现,返回的是“类名@hashcode”这样的格式的字符串。其中hashcode部分的值默认返回的是对象的“身份哈希值”(identity hash code)。jdk8u/jdk8u/jdk: 3dc438e0c8e1 src/share/classes/java/lang/Object.java

作者:RednaxelaFX
链接:https://www.zhihu.com/question/49631727/answer/120113928
来源:知乎
著作权归作者所有。商业转载请联系作者获得授权,非商业转载请注明出处。

/**
     * Returns a string representation of the object. In general, the
     * {@code toString} method returns a string that
     * "textually represents" this object. The result should
     * be a concise but informative representation that is easy for a
     * person to read.
     * It is recommended that all subclasses override this method.
     * <p>
     * The {@code toString} method for class {@code Object}
     * returns a string consisting of the name of the class of which the
     * object is an instance, the at-sign character `{@code @}', and
     * the unsigned hexadecimal representation of the hash code of the
     * object. In other words, this method returns a string equal to the
     * value of:
     * <blockquote>
     * <pre>
     * getClass().getName() + '@' + Integer.toHexString(hashCode())
     * </pre></blockquote>
     *
     * @return  a string representation of the object.
     */
    public String toString() {
        return getClass().getName() + "@" + Integer.toHexString(hashCode());
    }

jdk8u/jdk8u/jdk: 3dc438e0c8e1 src/share/classes/java/lang/System.java

/**
     * Returns the same hash code for the given object as
     * would be returned by the default method hashCode(),
     * whether or not the given object's class overrides
     * hashCode().
     * The hash code for the null reference is zero.
     *
     * @param x object for which the hashCode is to be calculated
     * @return  the hashCode
     * @since   JDK1.1
     */
    public static native int identityHashCode(Object x);

equals 和hashcode

hashcode:对象的初始地址的整数表示

Java中的对象是JVM在管理,JVM会在她认为合适的时候对对象进行移动,比如,在某些需要整理内存碎片的GC算法下发生的GC。此时,对象的地址会变动,但hashcode不会改变。

1.hashCode是为了提高在散列结构存储中查找的效率,在线性表中没有作用。

   2.一般一个类的对象如果会存储在HashTable,HashSet,HashMap等散列存储结构中,那么重写equals后最好也重写hashCode,否则会导致存储数据的不唯一性(存储了两个equals相等的数据)。而如果确定不会存储在这些散列结构中,则可以不重写hashCode。
   3.若两个对象equals返回true,则hashCode有必要也返回相同的int数。
   4.若两个对象equals返回false,则hashCode不一定返回不同的int数,但为不相等的对象生成不同hashCode值可以提高哈希表的性能。

   5.若两个对象hashCode返回相同int数,则equals不一定返回true。

   6.若两个对象hashCode返回不同int数,则equals一定返回false。

   7.同一对象在执行期间若已经存储在集合中,则不能修改影响hashCode值的相关信息,否则会导致内存泄露问题。

8.一般来说涉及到对象之间的比较大小就需要重写equals方法。

gc如果使用了复制算法,改变的对象的内存,那么对象的hashcode

内存地址改变,hashcode不会改变

hashCode 的常规协定是: 在 Java 应用程序执行期间,在同一对象上多次调用 hashCode 方法时,必须一致地返回相同的整数,前提是对象上 equals 比较中所用的信息没有被修改。从某一应用程序的一次执行到同一应用程序的另一次执行,该整数无需保持一致。 如果根据 equals(Object) 方法,两个对象是相等的,那么在两个对象中的每个对象上调用 hashCode 方法都必须生成相同的整数结果。

转载于:https://my.oschina.net/u/3421984/blog/3009825

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值