麻豆小视频在线观看_中文黄色一级片_久久久成人精品_成片免费观看视频大全_午夜精品久久久久久久99热浪潮_成人一区二区三区四区

首頁 > 學院 > 開發(fā)設計 > 正文

Java 中關于 unsaved-value 的問題

2019-11-18 13:16:29
字體:
來源:轉載
供稿:網(wǎng)友

  unsaved-value是表示一個對象是新的還是舊的,假如unsaved-value=none 那么就是新的,就會被insert到數(shù)據(jù)庫中,假如unsaved-value=any 就是說明對象是從數(shù)據(jù)庫中l(wèi)oad的,被update到數(shù)據(jù)庫中。
  
  我的問題是:unsaved-value是由我們來強制說明這個對象是新的還是舊的,那假如我把一個對象的unsaved-value設置為any,那我要new 一個對象,把他save到數(shù)據(jù)庫中,怎么做呢?我感覺這不是矛盾了嗎?主要是我們在寫配置文件的時候怎么能說一個對象就一定是new的還是load的?
  
  當你顯式的使用session.save()或者session.update()操作一個對象的時候,實際上是用不到unsaved-value的。某些情況下(父子表關聯(lián)保存),當你在程序中并沒有顯式的使用save或者update一個持久對象,那么Hibernate需要判定被操作的對象究竟是一個已經(jīng)持久化過的持久對象,是一個尚未被持久化過的內存臨時對象。例如:
  
  代碼:
  Session session = ...;
  Transaction tx = ...;
  
  Parent parent = (Parent) session.load(Parent.class, id);
  
  Child child = new Child();
  child.setParent(parent);
  child.setName("sun");
  
  parent.addChild(child);
  s.update(parent);
  
  s.flush();
  tx.commit();
  s.close();
  
  
  在上例中,程序并沒有顯式的session.save(child); 那么Hibernate需要知道child究竟是一個臨時對象,還是已經(jīng)在數(shù)據(jù)庫中有的持久對象。假如child是一個新創(chuàng)建的臨時對象(本例中就是這種情況),那么Hibernate應該自動產生session.save(child)這樣的操作,假如child是已經(jīng)在數(shù)據(jù)庫中有的持久對象,那么Hibernate應該自動產生session.update(child)這樣的操作。
  
  因此我們需要暗示一下Hibernate,究竟child對象應該對它自動save還是update。在上例中,顯然我們應該暗示Hibernate對child自動save,而不是自動update。那么Hibernate如何判定究竟對child是save還是update呢?它會取一下child的主鍵屬性 child.getId() ,這里假設id是 java.lang.Integer類型的。假如取到的Id值和hbm映射文件中指定的unsave-value相等,那么Hibernate認為child是新的內存臨時對象,發(fā)送save,假如不相等,那么Hibernate認為child是已經(jīng)持久過的對象,發(fā)送update。
  
  unsaved-value="null" (默認情況,適用于大多數(shù)對象類型主鍵 Integer/Long/String/...)
  
  當Hibernate取一下child的Id,取出來的是null(在上例中肯定取出來的是null),和unsaved-value設定值相等,發(fā)送save(child)
  
  當Hibernate取一下child的id,取出來的不是null,那么和unsaved-value設定值不相等,發(fā)送update(child)
  
  例如下面的情況:
  
  代碼:
  Session session = ...;
  Transaction tx = ...;
  
  Parent parent = (Parent) session.load(Parent.class, id);
  Child child = (Child) session.load(Child.class, childId);
  
  child.setParent(parent);
  child.setName("sun");
  
  parent.addChild(child);
  s.update(parent);
  
  s.flush();
  tx.commit();
  s.close();
  
  child已經(jīng)在數(shù)據(jù)庫中有了,是一個持久化的對象,不是新創(chuàng)建的,因此我們希望Hibernate發(fā)送update(child),在該例中,Hibernate取一下child.getId(),和unsave-value指定的null比對一下,發(fā)現(xiàn)不相等,那么發(fā)送update(child)。
  
  BTW: parent對象不需要操心,因為程序顯式的對parent有l(wèi)oad操作和update的操作,不需要Hibernate自己來判定究竟是save還是update了。我們要注重的只是child對象的操作。另外unsaved-value是定義在Child類的主鍵屬性中的。
  
  代碼:
  <class name="Child" table="child">
  <id column="id" name="id" type="integer" unsaved-value="null">
   <generator class="identity"/>
  </id>
  ...
  </class>
  
  假如主鍵屬性不是對象型,而是基本類型,如int/long/double/...,那么你需要指定一個數(shù)值型的unsaved-value,例如:
  
  代碼:
  unsaved-null="0"
  
  在此提醒大家,很多人以為對主鍵屬性定義為int/long,比定義為Integer/Long運行效率來得高,認為基本類型不需要進行對象的封裝和解構操作,因此喜歡把主鍵定義為int/long的。但實際上,Hibernate內部總是把主鍵轉換為對象型進行操作的,就算你定義為int/long型的,Hibernate內部也要進行一次對象構造操作,返回給你的時候,還要進行解構操作,效率可能反而低也說不定。因此大家一定要扭轉一個觀點,在Hibernate中,主鍵屬性定義為基本類型,并不能夠比定義為對象型效率來的高,而且也多了很多麻煩,因此建議大家使用對象型的Integer/Long定義主鍵。
  
  unsaved-value="none"和
  unsaved-value="any"
  
  主主要用在主鍵屬性不是通過Hibernate生成,而是程序自己setId()的時候。
  
  在這里多說一句,強烈建議使用Hibernate的id generator,或者你可以自己擴展Hibernate的id generator,非凡注重不要使用有實際含義的字段當做主鍵來用!例如用戶類User,很多人喜歡用用戶登陸名稱做為主鍵,這是一個很不好的習慣,當用戶類和其他實體類有關聯(lián)關系的時候,萬一你需要修改用戶登陸名稱,一改就需要改好幾張表中的數(shù)據(jù)。偶合性太高,而假如你使用無業(yè)務意義的id generator,那么修改用戶名稱,就只修改user表就行了。
  
  由這個問題引申出來,假如你嚴格按照這個原則來設計數(shù)據(jù)庫,那么你基本上是用不到手工來setId()的,你用Hibernate的id generator就OK了。因此你也不需要了解當
  
  unsaved-value="none"和
  unsaved-value="any"
  
  究竟有什么含義了。假如你非要用assigned不可,那么繼續(xù)解釋一下:
  
  unsaved-value="none" 的時候,由于不論主鍵屬性為任何值,都不可能為none,因此Hibernate總是對child對象發(fā)送update(child)
  
  unsaved-value="any" 的時候,由于不論主鍵屬性為任何值,都肯定為any,因此Hibernate總是對child對象發(fā)送save(child)
  
  大多數(shù)情況下,你可以避免使用assigned,只有當你使用復合主鍵的時候不得不手工setId(),這時候需要你自己考慮究竟怎么設置unsaved-value了,根據(jù)你自己的需要來定。
  
  BTW: Gavin King強烈不建議使用composite-id,強烈建議使用UserType。
  
  因此,假如你在系統(tǒng)設計的時候,遵循如下原則:
  
  1、使用Hibernate的id generator來生成無業(yè)務意義的主鍵,不使用有業(yè)務含義的字段做主鍵,不使用assigned。
  
  2、使用對象類型(String/Integer/Long/...)來做主鍵,而不使用基礎類型(int/long/...)做主鍵
  
  3、不使用composite-id來處理復合主鍵的情況,而使用UserType來處理該種情況。
  
  那么你永遠用的是unsaved-value="null" ,不可能用到any/none/..了。

發(fā)表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發(fā)表
主站蜘蛛池模板: 日韩2区 | 欧美大电影免费观看 | 亚洲不卡 | 久久蜜桃香蕉精品一区二区三区 | hd porn 4k video xhicial | 久久精品视频网址 | 国产在线看一区 | 欧美日韩在线中文 | a级黄色片视频 | 色综合777 | 久久久久久免费免费 | 最新中文字幕在线 | 毛片在线免费视频 | 激情在线视频 | 亚洲网站在线观看视频 | 久草在线高清 | 免费观看欧美一级片 | 欧美日韩国产一区二区三区在线观看 | 成人短视频在线播放 | 毛片视频网址 | 日本欧美一区二区三区视频麻豆 | 九九黄色 | 色天使中文字幕 | 亚洲第一页综合 | 日韩精品一区二区三区中文 | 手机黄色小视频 | 成人三级电影网 | 中文字幕亚洲视频 | 毛片免费看的 | 日韩精品中文字幕一区二区 | 一级片免费在线 | 精品久久久久久久久亚洲 | 国产精品自拍啪啪 | 亚洲情在线 | 免费欧美精品 | 国产精品久久久久久一区二区三区 | 久久一本日日摸夜夜添 | 欧美日韩精品一区二区三区不卡 | 国产精品亚洲精品日韩已方 | 懂色av懂色aⅴ精彩av | 免费国产人成网站 |