Hibernate中的對象狀態,及自動更新緣由

最近作項目時發現一個hibernate對象自動更新問題順便記錄一下:部分代碼以下;html

一、controller層java

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

三、打印出的日誌緩存

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=?

 

從以上三部分能夠看出,代碼中並未執行更新操做,然而日誌文件卻打印出了更新語句,因而搜索了一下緣由,感受下邊這篇文章很好便粘貼了下來。session

原文地址: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的運行機理。ui

一. 瞬時狀態(Transient)spa

由new命令開闢內存空間的java對象,它沒有持久化,沒有處於Session中,處於此狀態的對象叫瞬時對象。例:hibernate

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是不會幫你更新的)

在網上看到有人說遊離態的對象若是更改屬性也會自動更新,這個我就不作測試了,你們遇到能夠試一試。

若是想避免這種狀況:

一.在事務外面操做該對象;

二.克隆一個該對象。

相關文章
相關標籤/搜索