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

  对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;    
}

 

结果:


岁月无虞
1 声望0 粉丝

引用和评论

0 条评论