<programing in scala> 书中第28章讨论了对象相等性的问题,我觉得很有启发性。在这里从java语言的角度去解释,既为了自己能理解多一些,也可能对别人也有帮助。
相等性有以下的特点:
1. 自反的,即对任何非空的x,x.equals(x) 返回true
2. 对称, 即对于任何非空的x和y, x.equals(y) 当且仅当y.equals(x)返回true的时候返回true
3. 传递性, 即对于任何非空的x、y、z, x.equals(y) 返回true,且y.equals(z)返回true,则x.equals(z)返回true;
4. 对任何非空值x, x.equals(null)应返回false.
以下是一段典型的实现java 中euqlas,hashcode方法的代码
/** * */ package com.me.test; import java.util.HashSet; import java.util.Set; /** * @author Blues * */ public class PointApp1 { /** * @param args */ public static void main(String[] args) { Point a = new Point(1, 2); Point b = new ColoredPoint(1, 2, "red"); Set<Point> points = new HashSet<Point>(); points.add(a); System.out.println("set contains " + a + "?" + points.contains(a)); System.out.println("set contains " + b + "?" + points.contains(b)); //print false as b's Class is not Point. Point c = new Point(1, 1) { { this.setY(2); } }; System.out.println(a); //Point [x=1, y=2] System.out.println(c); //Point [x=1, y=2] System.out.println("set contains " + c + "?" + points.contains(c)); //false } } class Point { private int x; private int y; public Point(int x, int y) { super(); this.x = x; this.y = y; } //leave it only for child. protected void setX(int x) { this.x = x; } protected void setY(int y) { this.y = y; } @Override public int hashCode() { final int prime = 31; int result = 1; result = prime * result + x; result = prime * result + y; return result; } @Override public boolean equals(Object obj) { if (this == obj) return true; if (obj == null) return false; if (getClass() != obj.getClass()) return false; Point other = (Point) obj; if (x != other.x) return false; if (y != other.y) return false; return true; } @Override public String toString() { return "Point [x=" + x + ", y=" + y + "]"; } } class ColoredPoint extends Point { private String color; public ColoredPoint(int x, int y, String color) { super(x, y); this.color = color; } @Override public int hashCode() { final int prime = 31; int result = super.hashCode(); result = prime * result + ((color == null) ? 0 : color.hashCode()); return result; } @Override public boolean equals(Object obj) { if (this == obj) return true; if (!super.equals(obj)) return false; if (getClass() != obj.getClass()) return false; ColoredPoint other = (ColoredPoint) obj; if (color == null) { if (other.color != null) return false; } else if (!color.equals(other.color)) return false; return true; } @Override public String toString() { return "ColoredPoint [color=" + color + ", extends " + super.toString() + "]"; } }
程序运行的结果:
set contains Point [x=1, y=2]?true set contains ColoredPoint [color=red, extends Point [x=1, y=2]]?false Point [x=1, y=2] Point [x=1, y=2] set contains Point [x=1, y=2]?false
首先这样的实现是正确的,它满足上面提到的自反,对称,传递等性质。但是,考虑一下Point c,它的类型是继承了Point的匿名类,但它仅仅是为了修改y坐标,所以从逻辑上面来说,Point a和Point c都是在坐标系里面,且不带其他属性的(比如颜色)点(1, 2);所以认为它们相等也有一定的道理;
当然前提是,我们并不希望ColoredPoint b也和a相等;
在书中作者提到了一种方式; 前面的实现解决不了a和c相等的问题,是因为equals是比较两个对象静态的类,所以,如果能在运行时判断两个对象是否(可以)相等,就可以解决问题;
具体的实现如下:
/** * */ package com.me.test; import java.util.HashSet; import java.util.Set; /** * @author Blues * */ public class PointApp2 { /** * @param args */ public static void main(String[] args) { Point a = new Point(1, 2); Point b = new ColoredPoint(1, 2, "red"); Set<Point> points = new HashSet<Point>(); points.add(a); System.out.println("set contains " + a + "?" + points.contains(a)); System.out.println("set contains " + b + "?" + points.contains(b)); //print false as b's Class is not Point. Point c = new Point(1, 1) { { this.setY(2); } }; System.out.println(a); //Point [x=1, y=2] System.out.println(c); //Point [x=1, y=2] System.out.println("set contains " + c + "?" + points.contains(c)); //true } static class Point { private int x; private int y; public Point(int x, int y) { super(); this.x = x; this.y = y; } //leave it only for child. protected void setX(int x) { this.x = x; } protected void setY(int y) { this.y = y; } @Override public int hashCode() { final int prime = 31; int result = 1; result = prime * result + x; result = prime * result + y; return result; } @Override public boolean equals(Object obj) { if (this == obj) return true; if (obj == null) return false; // if (getClass() != obj.getClass()) // return false; Point other = (Point) obj; if(!other.canEquals(this)) { return false; } if (x != other.x) return false; if (y != other.y) return false; return true; } @Override public String toString() { return "Point [x=" + x + ", y=" + y + "]"; } protected boolean canEquals(Object obj) { return obj instanceof Point; } } static class ColoredPoint extends Point { private String color; public ColoredPoint(int x, int y, String color) { super(x, y); this.color = color; } @Override public int hashCode() { final int prime = 31; int result = super.hashCode(); result = prime * result + ((color == null) ? 0 : color.hashCode()); return result; } @Override public boolean equals(Object obj) { if (this == obj) return true; if (!super.equals(obj)) return false; // if (getClass() != obj.getClass()) // return false; ColoredPoint other = (ColoredPoint) obj; if(!other.canEquals(this)) { return false; } if (color == null) { if (other.color != null) return false; } else if (!color.equals(other.color)) return false; return true; } @Override public String toString() { return "ColoredPoint [color=" + color + ", extends " + super.toString() + "]"; } @Override protected boolean canEquals(Object obj) { return obj instanceof ColoredPoint; } } }
运行结果:
set contains Point [x=1, y=2]?true set contains ColoredPoint [color=red, extends Point [x=1, y=2]]?false Point [x=1, y=2] Point [x=1, y=2] set contains Point [x=1, y=2]?true
在Point类里面定义的一个canEquals 方法:当obj为Point的时候返回true;并且在equals方法里用
!that.canEquals(this)
代替
this.getClass() != that.getClass()
这里顺序很重要,必须要由that来作为caller,以保证对称性;
因为ColoredPoint override了canEquals,并且只在obj为ColoredPoint时才返回true,所以a和b是不相等的;但是匿名Point类没有覆盖这个类,所以a和c是相等的;
这个例子也许比较极端,很少会实用匿名Point,并且恰好又需要相等性判断;所以典型的相等性实现在大部分情况下都是可用的。
但是如果换个角度考虑,如果想要 ColoredPoint b 和a相等呢;虽然ColoredPoint多了个颜色属性,但在坐标系里面是同一个点;那么这种方法就可以很方便的修改,只需要不覆盖canEquals方法即可,而不用修改Point类的实现。 (但是不推荐这样做,因为这样实现ColoredPoint将违背传递性要求,考虑另外一个ColoredPoint d,坐标也为(1,2),但颜色是green,那么a equals b, a equals d, 但是b !equals d; 所以这里只是提到了一个潜在的优势)
另外值得一提的是,用scala来写这个例子要简短很多。
相关推荐
### Java代码编写一般性指导知识点详解 #### 一、命名规范 1. **类名**:首字母应大写,采用驼峰式命名法。例如:`ThisIsAClassName`。 2. **方法和变量**:首字母应小写,同样采用驼峰式命名法。例如:`...
- **平等性与不可变性**:探讨如何正确判断对象的相等性,以及如何创建不可变对象。 - **多线程与同步**:涉及线程安全问题,如何确保多线程环境下的数据一致性。 - **异常处理**:讨论Java中的异常处理机制,包括...
3. **测试字符串相等性**: - 错误做法:使用`compareTo`、`==`或不安全的方式检查字符串相等。 - 改进:推荐使用`equals()`方法,同时,比较空字符串时,使用`name.length() == 0`或`name.isEmpty()`更安全。 4....
实验报告的标题为“Java实验报告一java基础.doc”,主要涵盖了Java编程的基础...同时,这也强调了理论知识与实际操作相结合的重要性,实践中遇到的问题促使学习者去查阅文档,了解更多的API和函数,从而深化了学习。
为了使Person对象能在HashSet中正确工作,我们需要重写`hashCode()`和`equals()`方法,确保当两个Person对象的姓名和身份证号都相等时,它们被视为相等。然后,我们使用Iterator遍历HashSet并打印元素。 接着,我们...
在Java中,直接比较两个对象是否相等(或不等)应该使用`==`或`!=`,而非`equals()`方法。这里的关键在于,如果`addrName`为`null`,则调用`equals()`方法会抛出`NullPointerException`异常。因此,先进行`null`检查...
Java是一种广泛使用的面向对象的编程语言,以其跨平台性、高效性和丰富的库而著名。本文将深入探讨Java的基础知识,包括标识符、变量、常量、基本数据类型以及运算符等核心概念。 首先,让我们从Java标识符开始。在...
**Hashcode/Hash算法**:理解哈希码的作用及其与对象相等性的关系。哈希算法在数据结构如HashMap中非常重要。 **Linux安装Tomcat等应用**:熟悉在Linux环境下安装部署服务器环境的过程。 **Maven/SVN**:了解这两...
1. **基础语法**:题目可能要求考生写出特定的JAVA表达式,例如比较两个字符串是否相等,或者创建一个整数数组并初始化。 2. **面向对象**:考生可能需要设计一个类来满足特定的需求,比如创建一个银行账户类,包含...
- `equals()`:用于比较对象是否相等。 - `hashCode()`:返回对象的哈希码值。 - `toString()`:返回对象的字符串表示形式。 - `clone()`:实现`Cloneable`接口,用于浅拷贝对象。 - `Serializable`:实现此...
或者,可以采用超时机制,当哲学家等待一定时间仍无法获取资源时,会放弃当前请求并返回思考状态,以提高系统的响应性和吞吐量。 7. 结论哲学家就餐问题是一个典型的并发控制问题,它展示了多线程环境下的资源竞争...
Java 中的原子类是 java.util.concurrent.atomic 包下的类,之所以有原子性的共性,来源于 CAS。对原子类变量操作不会存在并发性问题,不需要使用同步进行并发控制。 锁机制在多线程中的应用 锁机制在多线程中的...
Interface解决了解决类之间的关系问题,可以定义一组方法签名,强制实现接口的类必须实现这些方法,增强了代码的抽象性和灵活性。 3. 对int数组进行排序的快速排序伪代码: ```java public int[] sort(int[] args) ...
在Java开发中,常常会遇到需要将一系列数据插入数据库的需求。但在实际操作过程中,为了避免重复数据的插入,我们需要实现一种机制来判断待插入的数据是否已存在于数据库中。这不仅可以提高数据的准确性,还能有效...
每个任务设计成独立的程序开发项目,旨在全面评估参赛者的代码实现、逻辑思考及问题解决能力。 #### 任务一:判断字符串是否为回文串 **知识点详解:** 1. **回文串概念**:回文串是一种特殊的字符串,其特性是...
在Java编程中,掌握基础知识至关重要。本文将通过四道基础题目,帮助你巩固和理解Java中的核心概念,包括对象引用...在学习Java的过程中,不断实践和思考这样的问题,能够有效巩固基础,为更高级的概念打下坚实的基础。
8. **植树问题**:在一个正方形的四个角各植一棵树,然后每边的中点再植一棵,这样任意两棵树之间的距离相等。 **第二组** 1. **下水道盖子问题**:圆形盖子不会因为重力而掉入洞中。 2. **中国汽车数量**:这个...
实验的目标是理解和掌握二叉树的链式存储结构,实现递归算法,并利用Java的泛型类来提高代码的可复用性。 首先,实验的第一个任务是计算二叉树的高度。二叉树的高度是指从根节点到最远叶子节点的最长路径上的边的...
【Java逻辑题】Java逻辑题主要锻炼的是程序员的逻辑思维能力,通过解决这些问题,开发者可以提升分析和解决问题的敏捷性。下面将详细解析几道题目: 1. **5升和6升水壶取3升水**:这是一道经典的逻辑题,通过一系列...