最近作項目時發現一個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是不會幫你更新的)
在網上看到有人說遊離態的對象若是更改屬性也會自動更新,這個我就不作測試了,你們遇到能夠試一試。
若是想避免這種狀況:
一.在事務外面操做該對象;
二.克隆一個該對象。