Hibernate中的对象状态,及自动更新原因

最近做项目时发现一个hibernate对象自动更新问题顺便记录一下:部分代码如下;

1、controller层

    public ModelAndView modifyAllEmp(
            @RequestParam(value = "type", required = false) String type,
            @RequestParam(value = "employeeId", required = false) String[] employeeId,
            @RequestParam(value = "terminalNum", required = false) String[] terminalNum,
            HttpSession session) throws ParseException{
        ModelMap modelMap = new ModelMap();
        if(!type.equals("1")&&(terminalNum==null||terminalNum.length==0)){
            modelMap.put("message", "请先选择下发终端");
            modelMap.put("result", "10000003");
            modelMap.put("success", false);
            new ModelAndView("jacksonJsonView", modelMap);
        }
        if(!type.equals("1")&&(employeeId==null||employeeId.length==0)){
            modelMap.put("message", "请先选择    下发员工");
            modelMap.put("result", "10000004");
            modelMap.put("success", false);
            new ModelAndView("jacksonJsonView", modelMap);
        }
        if(StringUtil.isBlank(type)){
            modelMap.put("message", "缺少参数type");
            modelMap.put("result", "10000002");
            modelMap.put("success", false);
            new ModelAndView("jacksonJsonView", modelMap);
        }
        modelMap.put("success", true);
        SysEmployee employee = (SysEmployee)session.getAttribute("sysEmployee");
        if(employee==null){
            modelMap.put("result", "99999999");
            modelMap.put("message","未登录");
            return new ModelAndView("jacksonJsonView", modelMap);
        }
        SimpleDateFormat formatter = new SimpleDateFormat("yyyy-mm-dd hh:MM:ss");
        if(type.equals("1")){
            ModifyLog modifyLog=modifyLogService.findByEntId(employee.getEnterpriseId());
            if(modifyLog!=null){
                 Date date1 = formatter.parse(modifyLog.getCreateTime());
                 Long dif = dateDiff(date1);
                if(dif<5){
                    modelMap.put("result", "10000004");
                    modelMap.put("message","请勿频繁点击下发按钮");
                    return new ModelAndView("jacksonJsonView", modelMap);
                }
                modifyLog.setCreateTime(formatter.format(new Date()));
                modifyLogService.update(modifyLog);
            }else{
                modifyLog=new ModifyLog();
                modifyLog.setCreateTime(formatter.format(new Date()));
                modifyLog.setEnterpriseId(employee.getEnterpriseId());
                modifyLogService.save(modifyLog);
            }
            sysEmployeeService.modifyAllEmp(employee);
        }else{
            sysEmployeeService.modifyEmpByTerminals(employee,terminalNum,employeeId);   
        }
        
        modelMap.put("result", "10000001");
        modelMap.put("message","成功");
        return new ModelAndView("jacksonJsonView", modelMap);
    }

 

2.service层

    public void modifyEmpByTerminals(SysEmployee emp,String[] terminalNums,String[] employeeIds) {
        if(StringUtil.isNotBlank(emp.getEnterpriseId())){
            for(String terminalNum:terminalNums){
                    if(StringUtil.isNotBlank(terminalNum)){
                        List<SysTerminal> ts= sysTerminalDao.findByTerNum(terminalNum);
                        SimpleDateFormat sdfSSS = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
                        String datetime = sdfSSS.format(new Date());
                        for(String empId:employeeIds){
                            SysEmployee employee=sysEmployeeDao.findById(empId);
                            if(employee!=null&&employee.getRole()!=null&!employee.getIsCard().equals("0")){
                                SysRole cc=roleDao.findById(employee.getRole());
                                if(cc!=null&&cc.getRoleType()!=null&&!cc.getRoleType().equals("01")){
                                    employee.setCreateTime(datetime);
                                    SysEmployeeModify smm=new SysEmployeeModify(employee,"1",ts.get(0).getFeNum(),terminalNum);
                                    Map<String, String> conditions = new HashMap<String, String>();
                                    conditions.put("terminalNum", terminalNum);
                                    conditions.put("jobNumber", employee.getJobNumber());
                                    conditions.put("chineseName", employee.getChineseName());
                                    conditions.put("password", employee.getPassword());
                                    conditions.put("cardNo", employee.getCardNo());
                                    conditions.put("time", "30");
                                    List<SysEmployeeModify> empMs=sysEmployeeModifyDao.findByCondition(conditions);
                                    if(empMs!=null&&empMs.size()>0){
                                        
                                    }else{
                                        sysEmployeeModifyDao.save(smm);
                                    }
                                }
                            }
                        }
                    }
            }
        }
    }

3、打印出的日志

Hibernate: insert into SYS_EMPLOYEE_MODIFY (CHINESE_NAME, CODE, create_Time, DEPT_ID, EMP_ID, ENTERPRISE_ID, FE_NUM, IMG, IS_CARD, JOB_NUMBER, MOBILE, PASSWORD, Role, SEX, STATUS, TERMINAL_NUM, TYPE, PK_ID) values (?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?)
Hibernate: update SYS_EMPLOYEE set create_Time=? where PK_ID=?

 

从以上三部分可以看出,代码中并未执行更新操作,然而日志文件却打印出了更新语句,于是搜索了一下原因,感觉下边这篇文章很好便粘贴了下来。

原文地址:https://www.cnblogs.com/xiaoda/p/3225617.html

Hibernate的对象有三种状态,分别为:瞬时状态 (Transient)、 持久化状态(Persistent)、游离状态(Detached)。对它的深入理解,才能更好的理解hibernate的运行机理。

一. 瞬时状态(Transient)

由new命令开辟内存空间的java对象,它没有持久化,没有处于Session中,处于此状态的对象叫瞬时对象。例:

Person person = new Person("韩梅梅", "女");

如果没有变量对该对象进行引用,它将被java虚拟机回收。

瞬时对象在内存孤立存在,它是携带信息的载体,不和数据库的数据有任何关联关系,在Hibernate中,可通过session的save()或saveOrUpdate()方法将瞬时对象与数据库相关联,并将数据对应的插入数据库中,此时该瞬时对象转变成持久化对象。

二.持久化状态(Persistent)

已经持久化,加入到了Session缓存中。

处于该状态的对象在数据库中具有对应的记录,并拥有一个持久化标识。如果是用hibernate的delete()方法,对应的持久对象就变成瞬时对象,因数据库中的对应数据已被删除,该对象不再与数据库的记录关联。

当一个session执行close()或clear()、evict()之后,持久对象变成脱管对象,此时该对象虽然具有数据库识别值,但它已不在HIbernate持久层的管理之下。

持久对象具有如下特点:

    1. 和session实例关联;

    2. 在数据库中有与之关联的记录。

三.游离状态(Detached)

当与某持久对象关联的session被关闭后,该持久对象转变为脱管对象。当脱管对象被重新关联到session上时,并再次转变成持久对象。

脱管对象拥有数据库的识别值,可通过update()、saveOrUpdate()等方法,转变成持久对象。

脱管对象具有如下特点:

    1.  本质上与瞬时对象相同,在没有任何变量引用它时,JVM会在适当的时候将它回收;

    2.  比瞬时对象多了一个数据库记录标识值。

 

下面我来说一下,很多朋友曾问我他并没有看到更新的代码,但是数据库里的数据却更新了,这个问题就是Hibernate导致的。

在一个事务中,如果为某持久化对象set了新的值,那么在提交事务时,Hibernate就会去比较你哪些持久化对象发生了变化,如果找到了,就会自动更新到数据库中。(注意,如果在事务外做的操作,Hibernate是不会帮你更新的)

在网上看到有人说游离态的对象如果更改属性也会自动更新,这个我就不做测试了,大家遇到可以试一试。

如果想避免这种情况:

一.在事务外面操作该对象;

二.克隆一个该对象。

 

Hibernate的对象有三种状态,分别为:瞬时状态 (Transient)、 持久化状态(Persistent)、游离状态(Detached)。对它的深入理解,才能更好的理解hibernate的运行机理。

一. 瞬时状态(Transient)

由new命令开辟内存空间的java对象,它没有持久化,没有处于Session中,处于此状态的对象叫瞬时对象。例:

Person person = new Person("韩梅梅", "女");

如果没有变量对该对象进行引用,它将被java虚拟机回收。

瞬时对象在内存孤立存在,它是携带信息的载体,不和数据库的数据有任何关联关系,在Hibernate中,可通过session的save()或saveOrUpdate()方法将瞬时对象与数据库相关联,并将数据对应的插入数据库中,此时该瞬时对象转变成持久化对象。

二.持久化状态(Persistent)

已经持久化,加入到了Session缓存中。

处于该状态的对象在数据库中具有对应的记录,并拥有一个持久化标识。如果是用hibernate的delete()方法,对应的持久对象就变成瞬时对象,因数据库中的对应数据已被删除,该对象不再与数据库的记录关联。

当一个session执行close()或clear()、evict()之后,持久对象变成脱管对象,此时该对象虽然具有数据库识别值,但它已不在HIbernate持久层的管理之下。

持久对象具有如下特点:

    1. 和session实例关联;

    2. 在数据库中有与之关联的记录。

三.游离状态(Detached)

当与某持久对象关联的session被关闭后,该持久对象转变为脱管对象。当脱管对象被重新关联到session上时,并再次转变成持久对象。

脱管对象拥有数据库的识别值,可通过update()、saveOrUpdate()等方法,转变成持久对象。

脱管对象具有如下特点:

    1.  本质上与瞬时对象相同,在没有任何变量引用它时,JVM会在适当的时候将它回收;

    2.  比瞬时对象多了一个数据库记录标识值。

 

下面我来说一下,很多朋友曾问我他并没有看到更新的代码,但是数据库里的数据却更新了,这个问题就是Hibernate导致的。

在一个事务中,如果为某持久化对象set了新的值,那么在提交事务时,Hibernate就会去比较你哪些持久化对象发生了变化,如果找到了,就会自动更新到数据库中。(注意,如果在事务外做的操作,Hibernate是不会帮你更新的)

在网上看到有人说游离态的对象如果更改属性也会自动更新,这个我就不做测试了,大家遇到可以试一试。

如果想避免这种情况:

一.在事务外面操作该对象;

二.克隆一个该对象。

posted @ 2018-08-30 14:05  根须  阅读(1231)  评论(0编辑  收藏  举报