Java代理 – >为什么代理对象与原始对象具有相同的hashCode

前端之家收集整理的这篇文章主要介绍了Java代理 – >为什么代理对象与原始对象具有相同的hashCode前端之家小编觉得挺不错的,现在分享给大家,也给大家做个参考。

我编写了这个测试类,我想知道为什么代理对象确实具有与原始对象相同的hashCode.有谁知道为什么?

public class Main {

public static void main(String[] args) {
    final Service realSubject = new Subject_A();
    final Service proxySubject = ProxyGenerator.makeProxy(Service.class,realSubject);
    final String hello = proxySubject.work("Hello");
    System.out.println("hello = " + hello);
    System.out.println("\n");
    System.out.println("realSubject: " + realSubject);
    System.out.println("proxySubject: " + proxySubject);
}
}
@H_301_8@

这是一个示例输出

in Subject_A#work: str = Hello
hello = Hello_DONE


realSubject: at.me.proxy.Subject_A@4f4a7090
proxySubject: at.me.proxy.Subject_A@4f4a7090
@H_301_8@
最佳答案
代理用于间接访问底层对象,就客户端代码而言,应隐藏代理的存在.

通常,此模式用于诸如spring和hibernate之类的框架中,以使用事务或安全功能来装饰对象.
鉴于上述情况,代理对象与hashcode(),equals()和toString()作为底层对象具有相同的输出是很自然的.

编辑

根据@Holger的更正进行更新

首先,您观察到的是toString()调用的相同输出,而不是hashcode().
通过代理实现equals()比初看起来更微妙一些.在equals()的典型实现中,根据equals contract会违反对属性

for any non-null reference values x and y,x.equals(y) should return true if and only if y.equals(x) returns true.

你有

// works since you delegate same instance of wrapped class to underyling object
proxy.equals(wrapped); // true
@H_301_8@

wrapped.equals(proxy); // false
@H_301_8@

由于:

 // proxy class != wrapped class 
 if (this.getClass() != obj.getClass()) {
        return false;
 }
@H_301_8@

正如@Holger建议的那样,包含相同底层实例的两个代理可以是相同的而不会违反对称性.

使代理等于包装实例的选项(反之亦然)可以是通过接口成员(getter)为包含对象相等性的状态实现equals,并将类与此接口进行比较.由于代理和底层对象都符合这个接口,因此它们是相同的.

猜你在找的Java相关文章