`

Java基础二: hashCode的作用简要说明,可能导致对象在集合中无法删除

阅读更多
import java.util.ArrayList;
import java.util.Collection;
import java.util.HashSet;

/**
 * hashCode的作用
 * 
 * @author 张明学
 * 
 */
public class HashCodeStudy {

	@SuppressWarnings("unchecked")
	public static void main(String[] args) {
		Point p1 = new Point(3, 4);
		Point p2 = new Point(4, 7);
		Point p3 = new Point(3, 4);

		System.out.println(p1.equals(p3));// 重写equals前:false
		System.out.println(p1.equals(p3));// 重写equals后:true

		Collection collection1 = new ArrayList();
		collection1.add(p1);
		collection1.add(p2);
		collection1.add(p3);
		collection1.add(p1);

		Collection collection2 = new HashSet();
		collection2.add(p1);
		collection2.add(p2);
		collection2.add(p3);
		collection2.add(p1);

		// 无论是否为重复放,大小为4
		System.out.println(collection1.size());
		// 对于set若重复(两个p1重复)放则放不进去,大小为3
		System.out.println(collection2.size());
		// 若重写equest方法并且重写了hashCode方法即x与y的值相同则让它们的equest和hashCode相等,
		// 这两个对象equest相等并放在同一块内存中,即p1与p3重复只能放进去一个。
		// **注意**:对于hashSet而言,并不是所有对象放在一块内存中,根据对象的hash值将它们的分到不同的内存块中
		// 若没有重写hashCode方法,若p1与p3的hashCode值不在同一块内存中,这们虽然它们的equest相同但它们不一定
		// 重复。(即p1与p3有可能能同放入hashSet中)
		// 一般情况我们会让两个对的equest方法相同时也会让它们的hashCode相同,就不会出现上述情况
		System.out.println(collection2.size());// 这里我们让poin类的equest与hashCode相同,故大小为2(p1与p3重复)

		// 由于对于hash类的集合是根据对象的hash值然后计算再放入内存中,若将对象放入hash类的集合中后修改了对象参入了
		// 计算hash值的属性值(如与x或y的值),即改变了对象的hash值则该对象有可能就删除不掉!!!因为hash值改变了
		// 删除的时候根据后来的hash值计算出该对象在内存中所存放的内存块(有可能与放入之前不是同一块内存)找该对象,
		// 就找不到了,故也删除不掉。如下例:
		p2.x = 24;// 修改了x的值也改变了p2的hashCode值
		collection2.remove(p2);// 删除p2
		System.out.println(collection2.size());// 大小还是为2与上面的大小一样,没有删除掉!!!

		// 上面的情况也就引起了Java中的内存泄露!!!

	}

}

 

public class Point {
	public int x;

	private int y;

	public Point(int x, int y) {
		this.x = x;
		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;
		final Point other = (Point) obj;
		if (x != other.x)
			return false;
		if (y != other.y)
			return false;
		return true;
	}

}

 

分享到:
评论

相关推荐

    Java容器简要介绍

    Java容器是Java编程中不可或缺的一部分,它们主要用于存储和管理对象,提供了一种高效且灵活的方式来组织和操作数据。本文将简要介绍Java容器的主要概念、API以及常用的容器类。 首先,Java容器分为两大主要类别:...

    Java语言与面向对象程序设计第05讲(语言基础类库和数组)

    Java语言与面向对象程序设计是学习Java编程的基础,这一部分主要涵盖了Java语言的诸多核心概念。在第05讲中,我们重点讨论了语言基础类库以及数组。 首先,回顾了之前章节的内容,包括Java程序的基本构成,如主类、...

    java面试大全-黑马

    例如,它可能包含了Java基础、面向对象编程、异常处理、集合框架、多线程、I/O流、网络编程、反射、内存管理等核心模块的简要介绍。 **第二章 JavaSE基础** 这部分深入探讨Java的面向对象特性,包括封装、继承、...

    Effective-Java-2nd-Edition-(May-2008).zip_effective java

    2. **重写equals()和hashCode()**:Bloch建议当重写equals()方法时,也应同时重写hashCode(),以确保对象在散列结构(如HashSet或HashMap)中的行为正确。 3. **枚举代替常量类**:书中推荐使用枚举类型替代传统的...

    Java最常见的面试题208道.docx

    Java编程语言在面试中常常涉及众多知识点,涵盖了基础语法、容器、多线程、反射、对象拷贝、JavaWeb、异常处理、网络编程以及框架等多个领域。以下是对这些主题的详细解析: 一、基础 1. JDK、JRE和JVM的区别:JDK...

    Java后端技术面试汇总.docx

    - **面向对象的特征**:继承、封装和多态是Java面向对象编程的三大特性,它们提供了结构化编程的基础。 - **final, finally, finalize**:final用于声明不可改变的变量或类;finally确保代码块在任何情况下都会被...

    hibernate_reference中文文档.pdf

    - **8.2 在集合中出现的依赖对象 (Collections of dependent objects)**:讨论如何映射集合中的依赖对象。 - **8.3 组件作为 Map 的索引 (Components as Map indices)**:解释如何将组件用作 Map 索引。 - **8.4 ...

    java面试问题汇总(非常全面)

    - `wait` 方法需要在 `synchronized` 块或方法中使用,并且会导致线程释放锁。 - `sleep` 方法是 Thread 类的方法,不会释放锁。 #### 14. final和finally,finalize的区别 - **final**:修饰类、方法或变量,表示...

    Java常用类别用法和进阶代码

    Java是世界上最流行的编程语言之一,尤其在企业级应用开发中占据主导地位。了解并熟练掌握Java的常用类库是每个Java开发者必须具备的基础。本文将深入探讨Java.lang包中的核心类,以及Java.util包中的一些关键组件。...

    hibernate3.5.6中文文档pdf格式

    - **在JTA环境下使用CurrentSessioncontext(当前Session上下文)管理**:介绍了如何在JTA环境中使用CurrentSessioncontext进行Session管理。 - **JMX部署**:说明了如何通过JMX进行部署。 - **第4章 持久化类...

    面试题大全(程序员cxuan编写).pdf

    这份“面试题大全”涵盖了Java编程语言的多个关键知识点,包括基础语法、对象与类、字符串操作、集合框架、并发处理、字符编码、网络协议、数据库以及SQL查询等。以下是对这些主题的详细阐述: 1. **Java基础知识**...

    Kotlin for android developers中文版

    - **映射对象到变量中**:讨论了如何将数据类中的对象映射到其他变量中。 #### 四、进阶主题 - **操作符重载** - **操作符表**:列出了Kotlin中可用的操作符及其对应的功能。 - **例子**:通过实例展示了如何在...

    hibernate_reference.pdf

    - **值类型的集合**:讨论如何处理集合中包含的值类型元素。 - **双向关联**:介绍如何实现实体间双向引用的映射。 - **使双向连起来**:详细说明如何确保双向关联能够正常工作。 **1.3 第三部分 - EventManager ...

    treemap treeset hashset hashmap 简要介绍

    在Java编程语言中,集合框架提供了多种数据结构来存储和操作数据,其中`TreeMap`、`TreeSet`、`HashSet`以及`HashMap`是最常用的数据结构之一。这些集合类各自有着独特的特性和应用场景,下面将对它们进行详细介绍。...

    EffectiveJava

    这两个方法的正确实现是保证类在集合中正确工作的基础。 2. **避免使用原始类型数组作为公共API**:使用List、Set和Map等集合类代替,因为它们提供了更多的功能,并且能自动处理容量扩展和安全性。 3. **使用枚举...

    Hibernate 3.6.0.Final Reference PDF 手册

    - **上下文相关的会话(Contextual Session)**:介绍如何在Web应用中使用上下文相关的会话管理。 #### 3. 配置 - **可编程的配置方式**:使用Java代码配置Hibernate。 - **获得SessionFactory**:介绍如何创建...

    Kotlin 编程三分钟入门

    以下是对Kotlin基础语法的简要介绍: 1. **变量定义**: 在Kotlin中,定义变量有两种方式:`val`用于声明不可变变量,而`var`用于声明可变变量。类型声明通常紧跟在变量名后面,用冒号`:`分隔。如果编译器可以从...

Global site tag (gtag.js) - Google Analytics