`
forestking
  • 浏览: 44041 次
  • 性别: Icon_minigender_1
  • 来自: 杭州
社区版块
存档分类
最新评论

Java 相等性思考

 
阅读更多

<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代码编写一般性指导

    ### Java代码编写一般性指导知识点详解 #### 一、命名规范 1. **类名**:首字母应大写,采用驼峰式命名法。例如:`ThisIsAClassName`。 2. **方法和变量**:首字母应小写,同样采用驼峰式命名法。例如:`...

    Practicla Java.pdf(简体中文版)

    - **平等性与不可变性**:探讨如何正确判断对象的相等性,以及如何创建不可变对象。 - **多线程与同步**:涉及线程安全问题,如何确保多线程环境下的数据一致性。 - **异常处理**:讨论Java中的异常处理机制,包括...

    Java编程最差实践

    3. **测试字符串相等性**: - 错误做法:使用`compareTo`、`==`或不安全的方式检查字符串相等。 - 改进:推荐使用`equals()`方法,同时,比较空字符串时,使用`name.length() == 0`或`name.isEmpty()`更安全。 4....

    Java实验报告一java基础.doc

    实验报告的标题为“Java实验报告一java基础.doc”,主要涵盖了Java编程的基础...同时,这也强调了理论知识与实际操作相结合的重要性,实践中遇到的问题促使学习者去查阅文档,了解更多的API和函数,从而深化了学习。

    实验05 Java集合.doc

    为了使Person对象能在HashSet中正确工作,我们需要重写`hashCode()`和`equals()`方法,确保当两个Person对象的姓名和身份证号都相等时,它们被视为相等。然后,我们使用Iterator遍历HashSet并打印元素。 接着,我们...

    简单实用java代码

    在Java中,直接比较两个对象是否相等(或不等)应该使用`==`或`!=`,而非`equals()`方法。这里的关键在于,如果`addrName`为`null`,则调用`equals()`方法会抛出`NullPointerException`异常。因此,先进行`null`检查...

    Java基础学习02.pdf

    Java是一种广泛使用的面向对象的编程语言,以其跨平台性、高效性和丰富的库而著名。本文将深入探讨Java的基础知识,包括标识符、变量、常量、基本数据类型以及运算符等核心概念。 首先,让我们从Java标识符开始。在...

    3年java面试经验

    **Hashcode/Hash算法**:理解哈希码的作用及其与对象相等性的关系。哈希算法在数据结构如HashMap中非常重要。 **Linux安装Tomcat等应用**:熟悉在Linux环境下安装部署服务器环境的过程。 **Maven/SVN**:了解这两...

    JAVA程序设计试题与答案

    1. **基础语法**:题目可能要求考生写出特定的JAVA表达式,例如比较两个字符串是否相等,或者创建一个整数数组并初始化。 2. **面向对象**:考生可能需要设计一个类来满足特定的需求,比如创建一个银行账户类,包含...

    java炸天组学习方式

    通过系统性地介绍Java基础知识及其应用,帮助读者建立起坚实的编程基础。文中提及的“Java炸天组学习方式”不仅强调了理论知识的重要性,更突出了实践操作的价值。 #### 二、环境搭建与基本概念 **1. Eclipse环境...

    新手入门写Java程序的三十个基本规则

    - `equals()`:用于比较对象是否相等。 - `hashCode()`:返回对象的哈希码值。 - `toString()`:返回对象的字符串表示形式。 - `clone()`:实现`Cloneable`接口,用于浅拷贝对象。 - `Serializable`:实现此...

    哲学家就餐问题实验报告java.docx

    或者,可以采用超时机制,当哲学家等待一定时间仍无法获取资源时,会放弃当前请求并返回思考状态,以提高系统的响应性和吞吐量。 7. 结论哲学家就餐问题是一个典型的并发控制问题,它展示了多线程环境下的资源竞争...

    算法Java经典.doc

    最后,将这些立方值相加,与原始数字比较,如果相等则该数是水仙花数。可以定义一个名为`shuixianhua`的函数来实现这一逻辑,函数接收一个整数参数,通过循环和条件语句找到所有水仙花数并返回。 最后是质因数分解...

    java进阶提高学习教程-14锁机制.pptx

    Java 中的原子类是 java.util.concurrent.atomic 包下的类,之所以有原子性的共性,来源于 CAS。对原子类变量操作不会存在并发性问题,不需要使用同步进行并发控制。 锁机制在多线程中的应用 锁机制在多线程中的...

    java笔试题目

    Interface解决了解决类之间的关系问题,可以定义一组方法签名,强制实现接口的类必须实现这些方法,增强了代码的抽象性和灵活性。 3. 对int数组进行排序的快速排序伪代码: ```java public int[] sort(int[] args) ...

    往数据库插入数据,相同的不插入

    在Java开发中,常常会遇到需要将一系列数据插入数据库的需求。但在实际操作过程中,为了避免重复数据的插入,我们需要实现一种机制来判断待插入的数据是否已存在于数据库中。这不仅可以提高数据的准确性,还能有效...

    七届全国信息技术应用水平大赛预赛试题\第七届全国信息技术应用水平大赛Java程序设计预赛主观题

    每个任务设计成独立的程序开发项目,旨在全面评估参赛者的代码实现、逻辑思考及问题解决能力。 #### 任务一:判断字符串是否为回文串 **知识点详解:** 1. **回文串概念**:回文串是一种特殊的字符串,其特性是...

    挑战4道Java试题

    在Java编程中,掌握基础知识至关重要。本文将通过四道基础题目,帮助你巩固和理解Java中的核心概念,包括对象引用...在学习Java的过程中,不断实践和思考这样的问题,能够有效巩固基础,为更高级的概念打下坚实的基础。

    java程序员招聘面试题

    8. **植树问题**:在一个正方形的四个角各植一棵树,然后每边的中点再植一棵,这样任意两棵树之间的距离相等。 **第二组** 1. **下水道盖子问题**:圆形盖子不会因为重力而掉入洞中。 2. **中国汽车数量**:这个...

    数据结构实验报告 二叉树.doc

    实验的目标是理解和掌握二叉树的链式存储结构,实现递归算法,并利用Java的泛型类来提高代码的可复用性。 首先,实验的第一个任务是计算二叉树的高度。二叉树的高度是指从根节点到最远叶子节点的最长路径上的边的...

Global site tag (gtag.js) - Google Analytics