Hibernate :A different object with the same identifier value was already associated with the session

这是Hibernate 通过session对象对数据库进行操作时遇到的错误,翻译过来就是:同一标识符值的另一个对象已经与会话关联。也就是说,在session中,不允许有两个主键相同的实体对象。举例说明:
Grade表中有gradeId和gradeName两个字段。
修改方法如下


    @Override
    public void update(Grade grade) {
        // TODO Auto-generated method stub
        //创建session对象
        Session session = SessionFactoryUtil.getSession();
        Transaction tx = session.beginTransaction();
        try {
            //通过gradeId获取表中的grade对象,判断要修改的对象是否存在
            Grade newGrade = session.get(Grade.class,grade.getGradeId());
            if(newGrade!=null)
            session.update(grade);
            tx.commit();
        } catch (Exception e) {
            // TODO: handle exception
            System.out.println(e.getMessage());
            tx.rollback();
        }

    }

这是修改方法的代码,在测试运行时就会抛出如标题的异常,原因是,在判断修改对象存不存在时,从表中获取的newGrade对象与session关联,在执行session.update(grade)时,grade对象与session关联,此时两个对象的主键gradeId
是相同的。

知道了报错的原因,解决方法就有了。

常用方法一:

            //通过gradeId获取表中的grade对象,判断要修改的对象是否存在
            Grade newGrade = session.get(Grade.class,grade.getGradeId());
            if(newGrade!=null)
            //将要修改的属性set到已和session关联的newGrade中
            newGrade.setGradeName(grade.getGradeName());
            //修改已和session关联的newGrade对象
            session.update(newGrade);
            tx.commit();

这个方法适合只修改部分属性的修改方法。

缺点是:要修改多少属性就要set多少次。

常用方法二:

    //通过gradeId获取表中的grade对象,判断要修改的对象是否存在
            Grade newGrade = session.get(Grade.class,grade.getGradeId());
            if(newGrade!=null)
            //清除已和session关联的对象
            session.clear();
            session.update(grade);
            tx.commit();

这个方法适用于修改实体对象所有属性的修改方法

缺点是:当只修改部分属性时,不能用该方法。

例如:某修改方法要求密码不能修改,传过来的对象中密码为空,此时用这个方法
会把密码修改为空。

以上只是和session关联的主键相同的对象在两个 以下,如果和session关联的主键相同的对象有多个时,就靠你自己相办法解决了,知道错误原因在这里,相信你可以解决的。

猜你喜欢

转载自blog.csdn.net/Fickle_actor/article/details/77962052