ORM基础——对象识别之重写对象equals()和hashCode()方法

在数据库中,主键值决定记录的唯一性。而在Java中,对象的识别可通过内存地址或重写equals()和hashCode()。默认的equals()基于内存地址比较,但在Hibernate中,相同session内的相同查询结果可能引用不同对象。因此,对比对象内容的正确做法是重写equals()和hashCode()方法。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

  对数据库而言,其识别一条记录唯一性的方式是根据主键值,如果手上有两条记录,它们拥有同样的主键值,则它们在数据库中代表同一个字段的记录。

  对Java而言,要识别两个对象是否为同一个对象有两种方式
      (1)内存地址识别(“= =”号识别);
      (2)根据equals()、hashCode()中的定义 (默认Object类中定义的equals(Object o)方法也是按内存地址来比较的)

           源码如下:
                 public boolean equals(Object obj){
                               return  (this==obj);
                       }

 

先探讨第一种Java的识别方式在Hibernate中该注意的地方,在Hibernate中,如果是在同一个session中根据相同查询所得到的相同记录,则它们会拥有相同的Java识别

//对象识别
		
		public void testObjectIndentifySession(){
			Configuration cfg=null;
			SessionFactory sf=null;
			Session session=null;
			Transaction ts=null;
			try {
				sf=HibernateUtil.getSessionFactory(); //sessionFactory的单态模式
				session=sf.getCurrentSession();//保证每个读写线程有唯一的session实例
				ts=session.beginTransaction();
                User u1=session.get(User.class, 2);
				User u2=session.get(User.class, 2);
				System.out.println(u1==u2);
                ts.commit();
			} catch (HibernateException e) {
				// TODO Auto-generated catch block
				e.printStackTrace();
				if(ts!=null){
					ts.rollback();
					}
				}finally{
				//session.close();
				//sf.close();
				}
		}

结果:

上面这个程序片段将会显示true的结果,表示u1与u2是参考至同一对象

如果是以下的情况则会显示false:

public void testObjectIndentifySession(){
			Configuration cfg=null;
			SessionFactory sf=null;
			Session session=null;
			Transaction ts=null;
			try {
				sf=HibernateUtil.getSessionFactory(); //sessionFactory的单态模式
				session=sf.getCurrentSession();//保证每个读写线程有唯一的session实例
				ts=session.beginTransaction();
					
				User u3=session.get(User.class, 3);
				session.evict(u3);
				User u4=session.get(User.class, 3);
			
				System.out.println(u3==u4);

				ts.commit();
			} catch (HibernateException e) {
				// TODO Auto-generated catch block
				e.printStackTrace();
				if(ts!=null){
					ts.rollback();
					}
				}finally{
				//session.close();
				//sf.close();
				}
		}

 

结果:

 

 

 

所以,使用==来比较两个对象的记录是否代表数据库中的同一条记录是不可行的。如果有必要比较通过查询后两个对象的内容是否相同,必须重写 equals()与hashCode()。

//manageUser.java
public void testObjectIndentifySession(){
			Configuration cfg=null;
			SessionFactory sf=null;
			Session session=null;
			Transaction ts=null;
			try {
				sf=HibernateUtil.getSessionFactory(); //sessionFactory的单态模式
				session=sf.getCurrentSession();//保证每个读写线程有唯一的session实例
				ts=session.beginTransaction();
        	User u3=session.get(User.class, 3);
				session.evict(u3);
				User u4=session.get(User.class, 3);
			
				System.out.println(u3==u4);
				System.out.println(u3.equals(u4));//因为是false,所以我们要重写equals()、hasCode()

            ts.commit();
			} catch (HibernateException e) {
				// TODO Auto-generated catch block
				e.printStackTrace();
				if(ts!=null){
					ts.rollback();
					}
				}finally{
				//session.close();
				//sf.close();
				}
		}

 

//User.java中重写equals、hashCode方法
public boolean equals(Object other) {        
		if (this == other) 
			return true;        
		if (!(other instanceof User)) 
			return false;       
		final User u = (User)other;        
		if (! name.equals(u.getName())) 
			return false;        
		if (!birthday.equals(u.getBirthday())) 
			return false;
		if (!(age==u.age)) 
		     return false;    
		if(! gender.equals(u.gender))
			return false;   
		return true;    
	}
	
	public int hashCode() { 
		int result;        
		result = name.hashCode();        
		result = 29 * result + getBirthday().hashCode();      
		return result;    
	}

 

结果:

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值