2.Methods Common to All Objects--Effective Java 2nd Ed学习笔记

Item 8: Obey the general contract when overriding equals

Equivalence relation(equals必须满足的性质):

Reflexive: For any non-null reference value x, x.equals(x) must return true.
Symmetric: For any non-null reference values x and y, x.equals(y) must return true if and only if y.equals(x) returns true.
Transitive: For any non-null reference values x, y, z, if x.equals(y) returns true and y.equals(z) returns true, then x.equals(z) must return true.
Consistent: For any non-null reference values x and y, multiple invocations of x.equals(y) consistently return true or consistently return false, provided no information used in equals comparisons on the objects is modified.
For any non-null reference value x, x.equals(null) must return false.

不满足对称性的例子:

// Broken - violates symmetry! public final class CaseInsensitiveString { private final String s; public CaseInsensitiveString(String s) { if (s == null) throw new NullPointerException(); this.s = s; } // Broken - violates symmetry! @Override public boolean equals(Object o) { if (o instanceof CaseInsensitiveString) return s.equalsIgnoreCase( ((CaseInsensitiveString) o).s); if (o instanceof String) // One-way interoperability! return s.equalsIgnoreCase((String) o); return false; } ... // Remainder omitted }

CaseInsensitiveString cis = new CaseInsensitiveString("Polish"); String s = "polish";

Once you’ve violated the equals contract, you simply don’t know how other objects will behave when confronted with your object.

为了消除该问题,必须避免让CaseInsensitiveString和String进行比较,代码如下:@Override public boolean equals(Object o) { return o instanceof CaseInsensitiveString && ((CaseInsensitiveString) o).s.equalsIgnoreCase(s); }

另一个例子:public class Point { private final int x; private final int y; public Point(int x, int y) { this.x = x; this.y = y; } @Override public boolean equals(Object o) { if (!(o instanceof Point)) return false; Point p = (Point)o; return p.x == x && p.y == y; } ... // Remainder omitted } 现在需要继承这个Point,并加上Color属性如下:public class ColorPoint extends Point { private final Color color; public ColorPoint(int x, int y, Color color) { super(x, y); this.color = color; } // Broken - violates symmetry! @Override public boolean equals(Object o) { if (!(o instanceof ColorPoint)) return false; return super.equals(o) && ((ColorPoint) o).color == color; } } 这样写违反了对称性(用一个Point与ColorPoint比较如:Point p = new Point(1, 2); ColorPoint cp = new ColorPoint(1, 2, Color.RED);)。改写equals如下:// Broken - violates transitivity! @Override public boolean equals(Object o) { if (!(o instanceof Point)) return false; // If o is a normal Point, do a color-blind comparison if (!(o instanceof ColorPoint)) return o.equals(this); // o is a ColorPoint; do a full comparison return super.equals(o) && ((ColorPoint)o).color == color; } 这样写避免了Point与ColorPoint比较时造成的对称性违反。但是违反了传递性:ColorPoint p1 = new ColorPoint(1, 2, Color.RED); Point p2 = new Point(1, 2); ColorPoint p3 = new ColorPoint(1, 2, Color.BLUE); //p1.equals(p2):true p2.equals(p3):true BUT p1.equals(p3):false事实上,在这种情况下无法实现满足这些性质的equals方法。There is no way to extend aninstantiable class and add a value component while preserving the equalscontract。这里的解决方法是:Item 16.Favorcomposition over inheritance.代码如下:// Adds a value component without violating the equals contract public class ColorPoint { private final Point point; private final Color color; public ColorPoint(int x, int y, Color color) { if (color == null) throw new NullPointerException(); point = new Point(x, y); this.color = color; } /** * Returns the point-view of this color point. */ public Point asPoint() { return point; } @Override public boolean equals(Object o) { if (!(o instanceof ColorPoint)) return false; ColorPoint cp = (ColorPoint) o; return cp.point.equals(point) && cp.color.equals(color); } ... // Remainder omitted }在Java类库中,java.sql.Timestampextends java.util.Date and adds a nanoseconds field.The equals implementationfor Timestamp does violate symmetry.将两者同时用在一个Collection中会导致错误。

recipe for a high-quality equals method:

1. Use the == operator to check if the argument is a reference to this object.

2. Use the instanceof operator to check if the argument has the correct type.

3. Cast the argument to the correct type.

4. For each “significant” field in the class, check if that field of the argument matches the corresponding field of this object.

primitive fields whose type is not float or double : ==

float/double fields:Float.compare/Double.compare(for Float.NaN, -0.0f)

object reference fields : equals

array:Arrays.equals

Some object reference fields may legitimately contain null.To avoid the possibility of a NullPointerException, use this idiom to compare such fields: (field == null ? o.field == null : field.equals(o.field))

5. When you are finished writing your equals method, ask yourself three questions: Is it symmetric? Is it transitive? Is it consistent?

你可能感兴趣的:(2.Methods Common to All Objects--Effective Java 2nd Ed学习笔记)