日韩av黄I国产麻豆传媒I国产91av视频在线观看I日韩一区二区三区在线看I美女国产在线I麻豆视频国产在线观看I成人黄色短片

歡迎訪問 生活随笔!

生活随笔

當前位置: 首頁 > 编程资源 > 编程问答 >内容正文

编程问答

session 中对象实例在不同事务中的状态

發布時間:2024/4/17 编程问答 35 豆豆
生活随笔 收集整理的這篇文章主要介紹了 session 中对象实例在不同事务中的状态 小編覺得挺不錯的,現在分享給大家,幫大家做個參考.

不同事務中執行hibernate query,則查詢出來的對象會在不同session中,或一個在托管態,一個在session中管理,所以是不同實例。

?

?

如果在同一事務中的話,則多次query出來的對象實際上是同一個實例,當你改變第一次query出來的對象時,之后你再query出來的實例就會反映第一次的變化。

同一事務下:

1.

swapTrade = swapTradeService.getTradeByLqtId(incomingTrade.getLqttradeid());

swapTrade = positionConverter.toSwapTrade(incomingTrade, swapTrade);

2.

SwapTrade persistedTrade = swapTradeService.getTradeByLqtId(swapTrade.getLqttradeid());

?

persistedTrade 會反映swapTrade的變化。

?

========================

http://www.iteye.com/topic/1113462

用過Hibernate的人都知道Hibernate最原始的使用Session方式(異常忽略):

?

獲取SessionFactory

打開Session

打開事務(可選)

執行操作

關閉事務(可選)

關閉Session

?

?

??? 當然還有另外一個方法getCurrentSession() 這個方法就是通過SessionContext來減少Session創建的。比如常用的ThreadLocalSessionContext:

?

Java代碼 ?
  • Session?current?=?existingSession(?factory?); ??
  • if?(current?==?null)?{ ??
  • ????current?=?buildOrObtainSession(); ??
  • ????current.getTransaction().registerSynchronization(?buildCleanupSynch()?); ??
  • ????if?(?needsWrapping(?current?)?)?{ ??
  • ????????current?=?wrap(?current?); ??
  • ????} ??
  • ????doBind(?current,?factory?); ??
  • } ??
  • return?current;??
  • Session current = existingSession( factory ); if (current == null) {current = buildOrObtainSession();current.getTransaction().registerSynchronization( buildCleanupSynch() );if ( needsWrapping( current ) ) {current = wrap( current );}doBind( current, factory ); } return current;

    ?

    ???? ?? currentSession()內部先從ThreadLocal中獲取Session。若不為null直接返回,若為null則openSession(...)一個Session并把這個Session對象綁定在ThreadLocal上。它就是通過在ThreadLocal中注冊綁定Session來確保線程中最多只有一個Session對象。

    ?

    ?

    上面的地球人都知道

    ==============================================================================

    ?

    Spring提供許多Template對各種底層ORM等進行集成,如JdbcTemplate、HibernateTemplate、JpaTemplate等同時也提供了相應的Dao模板類,如JdbcDaoSupport、HibernateDaoSupport、JpaDaoSupport等

    既然說Spring對Hibernate的集成,就得看HibernateTemplate和HibernateDaoSupport這兩個類。

    ???? 使用HibernateDaoSupport進行數據操作時常用兩種方式訪問Session:getSession()和HibernateCallback。

    注意:使用HibernateDaoSupport時候、如果再通過SessionFactory進行getCurrentSession()獲取Session的話就有可能出現問題了。因為Spring的Bean工廠把Hibernate的SessionFactory動了點小手腳 ~。就是前面說到的ThreadLocalSessionContext被Spring替換為 SpringSessionContext !

    這個SpringSessionContext的currentSession()方法核心如下:
    ? ? return (org.hibernate.classic.Session) SessionFactoryUtils.doGetSession(this.sessionFactory, false );
    ? ? // 這個false硬性規定doGetSession如果在線程中獲取不到Session不自動創建Session

    ?

    SessionFactoryUtils#doGetSession()是通過TransactionSynchronizationManager來獲取Session資源的,doGetSession()會通過 TransactionSynchronizationManager訪問線程資源、但是整個操作中沒有打開事務的話、此方法會拋出異常:
    ???????? org.hibernate.HibernateException: No Hibernate Session bound to thread, and configuration does not allow creation of non-transactional one here

    ?

    ??? 這個異常就是出現在SessionFactoryUtils#doGetSession()方法的最后:

    Java代碼 ?
  • if?(!allowCreate?&&?!isSessionTransactional(session,?sessionFactory))?{ ??
  • ????closeSession(session); ??
  • ????throw?new?IllegalStateException("No?Hibernate?Session?bound?to?thread,?"?+ ??
  • ????????"and?configuration?does?not?allow?creation?of?non-transactional?one?here"); ??
  • }??
  • if (!allowCreate && !isSessionTransactional(session, sessionFactory)) {closeSession(session);throw new IllegalStateException("No Hibernate Session bound to thread, " +"and configuration does not allow creation of non-transactional one here"); }?

    ?

    ?

    ????? ? getSession和getCurrentSession實現過程差不多,但是精簡了SessionContext那一段。并且可以設置allowCreate,如果設置為false,此方法應該和getCurrentSession同樣效果。

    ?

    ?

    =========================================================================

    ?

    ?

    了解了這些之后、我進行了一個HibernateTemplate和getSession之間的測試,主要就是測試

    單線程下多次重復請求創建Session與事務之間的關系、HibernateTemplate和getSession()兩者之間在功能實現和效率上有什么樣的不同。(多線程下卡的要死、不測試了)

    ?

    Dao層:testTS()方法——臨時創建用于測試的方法
    ??????????? 通過HibernateTemplate和直接getSession等方法獲取Session并保存。??????????
    ??????????? 打印已經保存的Session的狀態。
    ??????????? 打印已經保存的Session之間是否相同。
    ??????????? 打印SessionFactory的狀態記錄。
    Service層:testTS()方法——臨時創建用于測試的方法
    ??? ??? 內部代碼為執行三次請求:
    ???????? getDao().testTS();
    ???????? getDao().testTS();
    ???????? getDao().testTS();

    ?

    ?

    第一次測試

    ??? Service層的testTS(e)方法不打開事務。???
    ??? 單線程。

    ??? 使用HibernateTemplate。??

    結論:

    Service層沒有打開事務,但是每次操作CURD操作時,HibernateTemplate都會自動創建

    Transactional。同一線程獲取到的前后兩個Session之間互不相同。

    ? --------------Closed Transactional && Single-Thread--------------
    ? Last Session Status : false ?? //每次Session使用完自動關閉
    ? This Session Status : false
    ? Compared Session : false??

    ?

    ????? 測試數據擴大100倍并記時:
    ????? REQUEST[100]
    ????? Opened Session Count :300
    ????? Transaction Count :600?? //我沒有聲明打開事務啊!為什么它還為我打開事務
    ????? Connection Count :300
    ????? ---------------------------------------
    ????? TIME : 1719

    ?

    第二次測試

    ??? Service層的testTS(e)方法打開事務。???
    ??? 單線程。
    ??? 使用HibernateTemplate。

    結論:

    在事務邊界內獲取的Session由Spring管理、不必手動關閉

    雖然打開了事務,但是同一線程下同一事務邊界內前后獲取的兩個Session仍然不同! 這是為什么???

    后面會繼續剖析HibernateTemplate源碼,給出解釋

    ? --------------Open Transactional && Single-Thread--------------
    ? Last Session Status : Open ?? //每次Session使用完之后未關閉
    ? This Session Status : Open
    ? Compared Session : false??? //同一線程同一個事務前后獲取的兩個Session還不同!!!!!為什么?????

    ???

    ??????? 測試數據擴大100倍并記時:
    ??????? REQUEST[100]
    ??????? Opened Session Count :100
    ??????? Transaction Count :200
    ??????? Connection Count :100
    ??????? ---------------------------------------
    ??????? TIME : 1719

    ?

    第三次測試

    ??? Service層的testTS(e)方法不打開事務。???
    ??? 單線程。
    ??? 使用getSession()。

    可以看到getSession()由于缺少Spring的支持,在無事務環境下。

    它為每個CURD請求創建了一個Session。并且更讓人震驚的是,它好像為每個Session都創建了新

    Connection 。這些Session使用之后,它也并不關閉。

    ? --------------Closed Transactional && Single-Thread--------------
    ? Last Session Status : Open
    ? This Session Status : Open
    ? Compared Session : false

    ???

    ???? 測試數據擴大100倍并記時,令人極其震驚且極其坑爹的一幕出現了!我親眼看到程序以一種 肉眼可見的速度

    慢騰騰的執行一次又一次的循環,它竟然執行了一分鐘!!!:
    ??????? REQUEST[100]
    ??????? Opened Session Count :300
    ??????? Transaction Count :0
    ??????? Connection Count :300
    ??????? ---------------------------------------
    ??????? TIME : 53844

    ?

    第四次測試

    ??? Service層的testTS(e)方法打開事務。???
    ??? 單線程。

    ??? 使用getSession() 。

    這個最容易讓人理解、由于同一線程且在同一事務邊界內,前后兩個Session相同。并且也不會重復創建

    Connection。

    ? --------------Open Transactional && Single-Thread--------------
    ? Last Session Status : Open ?? //每次Session使用完之后未關閉
    ? This Session Status : Open
    ? Compared Session : True ?? //同一線程前后獲取的兩個Session相同

    ???????? 測試數據擴大100倍并記時:
    ???????? REQUEST[100]
    ???????? Opened Session Count :100
    ???????? Transaction Count :200
    ???????? Connection Count :100
    ???????? ---------------------------------------
    ???????? TIME : 1204

    ?

    ?

    總結:

    ????? getSession()在事務邊界內會通過TransactionSynchronizationManager獲取Session資源,同一線程內它不會重復創建Connection , 那些獲取到的Session()不需要手動關閉。但是在無聲明式事務環境下,它就會表現出極其坑爹的狀況,它反復獲取Connection,也不關閉Session。非常消耗資源

    ?

    ????? HibernateTemplate有一個安全且高效的Session環境,它的CRUD都是位于HibernateCallback內部,如果它的CRUD并沒有位于事務之中,它會自己創建一個事務(Spring集成Hibernate時,所有的資源都是由TransactionSynchronizationManager管理的 )。同一個線程中它只需要一個Connection。Session也會在事務邊界處自動關閉,程序員不需要關注此事(這個事務邊界可能是@Transactional顯式聲明的也可能是HibernateTemplate#doExecute隱式聲明的)。

    ?

    ???? 在同一個事務邊界內,兩個HibernateTemplate操作內部的Session互也不相同這個問題可以在HibernateTemplate源碼中找到答案:

    Java代碼 ?
  • protected?<T>?T?doExecute(HibernateCallback<T>?action,?boolean?enforceNewSession,?boolean?enforceNativeSession) ??
  • ????????throws?DataAccessException?{ ??
  • ????...... ??
  • ????Session?session?=?(enforceNewSession??? ??
  • ????????????????????????SessionFactoryUtils.getNewSession(getSessionFactory(),?getEntityInterceptor())?:?getSession()); ??
  • ????boolean?existingTransaction?=?(!enforceNewSession?&& ??
  • ????????????(!isAllowCreate()?||?SessionFactoryUtils.isSessionTransactional(session,?getSessionFactory()))); ??
  • ????if?(existingTransaction)?{ ??
  • ????????logger.debug("Found?thread-bound?Session?for?HibernateTemplate"); ??
  • ????} ??
  • ??
  • ????FlushMode?previousFlushMode?=?null; ??
  • ????try?{ ??
  • ????????previousFlushMode?=?applyFlushMode(session,?existingTransaction); ??
  • ????????enableFilters(session); ??
  • ????????Session?sessionToExpose?= ??
  • ????????????????(enforceNativeSession?||?isExposeNativeSession()??? ??
  • ????????????????????????????????session?:?createSessionProxy(session)); ??
  • ????????T?result?=?action.doInHibernate(sessionToExpose); ??
  • ????????flushIfNecessary(session,?existingTransaction); ??
  • ????????return?result; ??
  • ????...... ??
  • }??
  • protected <T> T doExecute(HibernateCallback<T> action, boolean enforceNewSession, boolean enforceNativeSession)throws DataAccessException {......Session session = (enforceNewSession ? SessionFactoryUtils.getNewSession(getSessionFactory(), getEntityInterceptor()) : getSession());boolean existingTransaction = (!enforceNewSession &&(!isAllowCreate() || SessionFactoryUtils.isSessionTransactional(session, getSessionFactory())));if (existingTransaction) {logger.debug("Found thread-bound Session for HibernateTemplate");}FlushMode previousFlushMode = null;try {previousFlushMode = applyFlushMode(session, existingTransaction);enableFilters(session);Session sessionToExpose =(enforceNativeSession || isExposeNativeSession() ? session : createSessionProxy(session));T result = action.doInHibernate(sessionToExpose);flushIfNecessary(session, existingTransaction);return result;...... } Java代碼 ?
  • 通過 ??
  • Session?sessionToExpose?= ??
  • ????????????????(enforceNativeSession?||?isExposeNativeSession()??? ??
  • ????????????????????????????????session?:?createSessionProxy(session)); ??
  • 紅色部分可以看到HibernateTemplate中獲取的Session不是原生的,而是代理的。那個代理類是一個比較簡單的內部類,源代碼位于HibernateTemplate類文件最下部分。每次HibernateTemplate#doExecute執行時除非聲明不使用代理類,Spring都會使用線程中的Session資源來創建代理。但是這個代理類的創建對性能的影響微不足道了,Spring這樣做肯定有它的道理。??
  • 通過 Session sessionToExpose =(enforceNativeSession || isExposeNativeSession() ? session : createSessionProxy(session)); 紅色部分可以看到HibernateTemplate中獲取的Session不是原生的,而是代理的。那個代理類是一個比較簡單的內部類,源代碼位于HibernateTemplate類文件最下部分。每次HibernateTemplate#doExecute執行時除非聲明不使用代理類,Spring都會使用線程中的Session資源來創建代理。但是這個代理類的創建對性能的影響微不足道了,Spring這樣做肯定有它的道理。

    ?

    ?

    ====================================

    http://www.kuqin.com/docs/hibernate/transactions.html

    第?10?章?事務和并行(Transactions And Concurrency)

    Hibernate本身并不是數據庫,它只是一個輕量級的對象-關系數據庫映射(object-relational)工具。它的事務交由底層的數據庫連接管理,如果數據庫連接有JTA的支持,那么在Session中進行的操作將是整個原子性JTA事務的一部分。Hibernate可以看作是添加了面向對象語義的JDBC瘦適配器(thin adapter)。

    10.1.?配置,會話和工廠(Configurations, Sessions and Factories)

    SessionFactory的創建需要耗費大量資源,它是線程安全(threadsafe)的對象,在應用中它被所有線程共享。而Session的創建耗費資源很少,它不是線程安全的對象,對于一個簡單商業過程(business process),它應該只被使用一次,然后被丟棄。舉例來說,當Hibernate在基于servlet的應用中,servlet能夠以下面的方式得到SessionFactory。

    SessionFactory sf = (SessionFactory)getServletContext().getAttribute("my.session.factory");

    每次調用SessionFactory的service方法能夠生成一個新的Session對象,然后調用Session的flush(),調用commit()提交它的連接,調用close()關閉它,最終丟棄它。(SessionFactory可能被保存在JNDI或者一個靜態的單例(Singleton)輔助變量中。)

    在無狀態的session bean中,可以同樣使用類似的方法。bean在setSessionContext()中得到SessionFactory的實例,每個商業方法會生成一個Session對象,調用它的flush()和close(),當然,應用不應該commit()connection. (把它留給JTA.在容器管理的事務中,數據庫連接會自動完成事務。)

    我們用上述方法使用Hibernate 的Transaction API,對Transaction執行一次commit()會把所有狀態同步,把底層的數據庫連接提交(對JTA 事務會特殊處理。)

    這里需要理解flush()的含義。 flush()將持久化存儲與內存中的變化進行同步,但不是將內存的變化與持久化存儲進行同步。注意對所有的Hibernate JDBD 連接/事務來說,其隔離級別將施加于所有的Hibernate執行的操作之上!

    接下來的幾小節將討論利用版本化的方法來確保事務原子性,這些“高級”方法需要小心使用。

    10.2.?線程和連接(Threads and connections)

    在創建Hibernate會話(Session)時,你應該留意以下的實踐(practices):

    • 對于一個數據庫連接,不要創建一個以上的Session或Transaction。

    • 在對于一個數據庫連接、一個事務使用多個Session時,你尤其需要格外地小心。Session對象會記錄下調入數據更新的情況,所以另一個Session對象可能會遇到過時的數據。

    • Session不是線程安全的。決不要在兩個并發的線程中訪問同一個Session。一個Session一般只對應一批需要一次性完成的單元操作!

    10.3.?考慮對象辨別

    程序可能在兩批單元操作中并發訪問同一個對象的持久化狀態。不管怎樣,持久化類的一個實例不可能在兩個Session中共享。所以有兩種不同的辨別方式:

    數據庫辨別

    foo.getId().equals( bar.getId() )

    JVM 辨別

    foo==bar

    對于依附于某個特定Session的對象,兩種辨別方式是等價的。然而,當程序可能在兩個不同的session中并發訪問“同一個”(持久化辨別)商業對象時,兩個實例(對于JVM辨別來說)卻可能是“不同”的。

    這種方式把關于并發的頭疼問題留給了Hibernate和數據庫。程序不需要對任何商業對象進行同步,只要程序堅持每個Session一個線程,或者對象辨別的策略(在一個Session重,程序可以安全的使用==來比較對象)。

    10.4.?樂觀并發控制(Optimistic concurrency control)

    許多商業過程需要一系列與用戶進行交互的過程,數據庫訪問穿插在這些過程中。對于web和企業應用來說,跨一個用戶交互過程的數據事務是不可接受的。

    維護各商業事務間的隔離(isolocation)就成為應用層的部分責任,我們把這種過程稱為長時間運行的應用事務(application transaction)。單一的應用事務可能跨越多個數據庫事務。如果這些數據庫事務中只有一個(最后一個)保存了被修改的數據,其他事務只是簡單地讀數據,則這個應用事務就是原子性的。

    唯一滿足高并發性以及高可擴展性的方法是使用帶有版本化的樂觀并發控制。Hibernate為使用樂觀并發控制的代碼提供了三種可能的方法。

    10.4.1.?使用長生命周期帶有自動版本化的會話

    在整個商業過程中使用一個單獨的Session實例以及它的持久化實例,這個Session使用帶有版本化的樂觀鎖定機制,來確保多個數據庫事務對于應用來說只是一個邏輯上的事務。在等待用戶交互時,Session斷開與數據庫的連接。這個方法從數據庫訪問方面來看是最有效的,應用不需要關心對自己的版本檢查或是重新與不需要序列化(transient)的實例進行關聯。

    在整個應用事務中,使用單一的Session 實例和它的持久化實例。

    Session 使用帶有版本化的樂觀鎖定來保證多個數據庫事務對程序來說就如同是單一的邏輯應用事務。在等待用戶交互的時候,Session 脫離所有的底層JDBC連接。對于數據庫訪問來說,這種方法是最高效的。程序自己不需要關心版本檢查或者把已經脫離session的實例重新關聯到session。

    // foo is an instance loaded earlier by the Session session.reconnect(); foo.setProperty("bar"); session.flush(); session.connection().commit(); session.disconnect();

    foo對象仍然知道是哪個Session把自己裝載的。 只要Session 擁有一個JDBC連接,我們可以把對象的更改提交。

    如果我們的 Session 太大,以至于在用戶思考的時間內無法保存住,這種模式就會出現問題。比如,HttpSession應該保持盡量小。因為Session也持有(必須的)第一級緩存,包含所有被裝載的對象,我們只能在很少的request/response周期中使用這一策略。這種少用是被鼓勵的,因為Session 很快就會出現過時的數據。

    10.4.2.?使用帶有自動版本化的多個會話

    每個與持久化存儲的交互出現在一個新的Session中,在每次與數據庫的交互中,使用相同的持久化實例。應用操作那些從其它Session調入的已經脫離session的實例的狀態,通過使用Session.update()或者Session.saveOrUpdate()來重新建立與它們的關聯。

    // foo is an instance loaded by a previous Session foo.setProperty("bar"); session = factory.openSession(); session.saveOrUpdate(foo); session.flush(); session.connection().commit(); session.close();

    你也可以調用lock()而非update(),如果你確信對象沒有被修改過,可以使用LockMode.READ(進行一次版本檢查,而跳過所有的緩存)。

    10.4.3.?應用程序自己進行版本檢查

    每當一個新的Session中與數據庫出現交互的時候,這個session會在操作持久化實例前重新把它們從數據庫中裝載進來。我們現在所說的方式就是你的應用程序自己使用版本檢查來確保應用事務的隔離性。(當然,Hibernate仍會為你更新版本號)。從數據庫訪問方面來看,這種方法是最沒有效率的,與entity EJB方式類似。

    // foo is an instance loaded by a previous Session session = factory.openSession(); int oldVersion = foo.getVersion(); session.load( foo, foo.getKey() ); if ( oldVersion!=foo.getVersion ) throw new StaleObjectStateException(); foo.setProperty("bar"); session.flush(); session.connection().commit(); session.close();

    當然,如果在低數據并行(low-data-concurrency)的環境中,并不需要版本檢查,你仍可以使用這個方法,只需要忽略版本檢查。

    10.5.?會話斷開連接(Session disconnection)

    The first approach described above is to maintain a single Session for a whole business process thats spans user think time. (For example, a servlet might keep a Session in the user's HttpSession.) For performance reasons you should

    上面提到的第一種方法是對于對一個用戶的一次登錄產生的整個商業過程維護一個Session。(舉例來說,servlet有可能會在用戶的HttpSession中保留一個Session)。為性能考慮,你必須

  • 提交Transaction(或者JDBC連接),然后

  • (在等待用戶操作前,)斷開Session與JDBC連接。

  • Session.disconnect()方法會斷開會話與JDBC的連接,把連接返還給連接池(除非是你自己提供這個連接的)。

    Session.reconnect()方法會得到一個新的連接(你也可以自己提供一個),重新開始會話。在重新連接后,你可以通過對任何可能被其它事務更新的對象調用Session.lock()方法,來強迫對你沒有更新的數據進行版本檢查。你不需要對正在更新的數據調用lock()。

    這是一個例子:

    SessionFactory sessions; List fooList; Bar bar; .... Session s = sessions.openSession();Transaction tx = null; try {tx = s.beginTransaction();fooList = s.find("select foo from eg.Foo foo where foo.Date = current date"// uses db2 date function);bar = (Bar) s.create(Bar.class);tx.commit(); } catch (Exception e) {if (tx!=null) tx.rollback();s.close();throw e; } s.disconnect();

    接下來:

    s.reconnect();try {tx = s.beginTransaction();bar.setFooTable( new HashMap() );Iterator iter = fooList.iterator();while ( iter.hasNext() ) {Foo foo = (Foo) iter.next();s.lock(foo, LockMode.READ); //check that foo isn't stalebar.getFooTable().put( foo.getName(), foo );}tx.commit(); } catch (Exception e) {if (tx!=null) tx.rollback();throw e; } finally {s.close(); }

    從上面的例子可以看到Transaction和Session之間是多對一的關系。一個Session表示了應用程序與數據庫之間的一個對話,Transaction把這個對話分隔成一個個在數據庫級別具有原子性的單元。

    10.6.?悲觀鎖定(Pessimistic Locking)

    用戶不需要在鎖定策略上花費過多時間,通常我們可以對JDBC連接選定一種隔離級別(isolationn level),然后讓數據庫完成所有的工作。高級用戶可能希望得到悲觀鎖定或者在新的事務開始時重新得到鎖。

    Hibernate一直都會使用數據庫的鎖定機制,而不會在內存中鎖定對象。

    LockMode類定義了Hibernate需要的不同的鎖級別。鎖由以下的機制得到:

    • LockMode.WRITE在Hibernate更新或插入一行數據時自動得到。

    • LockMode.UPGRADE在用戶通過SELECT ... FOR UPDATE這樣的特定請求得到,需要數據庫支持這種語法。

    • LockMode.UPGRADE_NOWAIT在用戶通過SELECT ... FOR UPDATE NOWAIT這樣的特定請求在Oracle數據庫環境下得到。

    • LockMode.READ在Hibernate在不斷讀(Repeatable Read)和序列化(Serializable)的隔離級別下讀取數據時得到。也可以通過用戶的明確請求重新獲得。

    • LockMode.NONE表示沒有鎖。所有對象在Transaction結束時會切換到這種鎖模式,通過調用update()或者saveOrUpdate()與會話進行關聯的對象,開始時也會在這種鎖模式。

    “明確的用戶請求”會以下的幾種方式出現:

    • 調用Session.load(),指定一種LockMode。

    • 調用Session.lock()。

    • 調用Query.setLockMode()。

    如果在調用Session.load()時指定了UPGRADE或者UPGRADE_NOWAIT,并且請求的對象還沒有被會話調入,那么這個對象會以SELECT ... FOR UPDATE的方式調入。如果調用load()在一個已經調入的對象,并且這個對象調入時的鎖級別沒有請求時來得嚴格,Hibernate會對這個對象調用lock()。

    Session.lock()會執行版本號檢查的特定的鎖模式是:READ,UPGRADE或者UPGRADE_NOWAIT。(在UPGRADE或者UPGRADE_NOWAIT,SELECT ... FOR UPGRADE使用的情況下。)

    如果數據庫不支持所請求的鎖模式,Hibernate將會選擇一種合適的受支持的鎖模式替換(而不是拋出一個異常)。這確保了應用具有可移植性。

    ?

    ============http://www.redsaga.com/hibernate-ref/3.x/zh-cn/html/objectstate.html

    第?10?章?與對象共事

    Hibernate是完整的對象/關系映射解決方案,它提供了對象狀態管理(state management)的功能,使開發者不再需要理會底層數據庫系統的細節。 也就是說,相對于常見的JDBC/SQL持久層方案中需要管理SQL語句,Hibernate采用了更自然的面向對象的視角來持久化Java應用中的數據。

    換句話說,使用Hibernate的開發者應該總是關注對象的狀態(state),不必考慮SQL語句的執行。 這部分細節已經由Hibernate掌管妥當,只有開發者在進行系統性能調優的時候才需要進行了解。

    10.1.?Hibernate對象狀態(object states)

    Hibernate定義并支持下列對象狀態(state):

    • 瞬時(Transient) - 由new操作符創建,且尚未與Hibernate Session 關聯的對象被認定為瞬時(Transient)的。瞬時(Transient)對象不會被持久化到數據庫中,也不會被賦予持久化標識(identifier)。 如果程序中沒有保持對瞬時(Transient)對象的引用,它會被垃圾回收器(garbage collector)銷毀。 使用Hibernate Session可以將其變為持久(Persistent)狀態。(Hibernate會自動執行必要的SQL語句)

    • 持久(Persistent) - 持久(Persistent)的實例在數據庫中有對應的記錄,并擁有一個持久化標識(identifier)。 持久(Persistent)的實例可能是剛被保存的,或剛被加載的,無論哪一種,按定義對象都僅在相關聯的Session生命周期內的保持這種狀態。 Hibernate會檢測到處于持久(Persistent)狀態的對象的任何改動,在當前操作單元(unit of work)執行完畢時將對象數據(state)與數據庫同步(synchronize)。 開發者不需要手動執行UPDATE。將對象從持久(Persistent)狀態變成瞬時(Transient)狀態同樣也不需要手動執行DELETE語句。

    • 脫管(Detached) - 與持久(Persistent)對象關聯的Session被關閉后,對象就變為脫管(Detached)的。 對脫管(Detached)對象的引用依然有效,對象可繼續被修改。脫管(Detached)對象如果重新關聯到某個新的Session上, 會再次轉變為持久(Persistent)的(Detached其間的改動將被持久化到數據庫)。 這個功能使得一種編程模型,即中間會給用戶思考時間(user think-time)的長時間運行的操作單元(unit of work)的編程模型成為可能。 我們稱之為應用程序事務,即從用戶觀點看是一個操作單元(unit of work)。

    接下來我們來細致的討論下狀態(states)及狀態間的轉換(state transitions)(以及觸發狀態轉換的Hibernate方法)。

    10.2.?使對象持久化

    Hibernate認為持久化類(persistent class)新實例化的對象是瞬時(Transient)的。 我們可將瞬時(Transient)對象與session關聯而變為持久(Persistent)的。

    DomesticCat fritz = new DomesticCat(); fritz.setColor(Color.GINGER); fritz.setSex('M'); fritz.setName("Fritz"); Long generatedId = (Long) sess.save(fritz);

    如果Cat的持久化標識(identifier)是generated類型的, 那么該標識(identifier)會自動在save()被調用時產生并分配給cat。 如果Cat的持久化標識(identifier)是assigned類型的,或是一個復合主鍵(composite key), 那么該標識(identifier)應當在調用save()之前手動賦予給cat。 你也可以按照EJB3 early draft中定義的語義,使用persist()替代save()。

    此外,你可以用一個重載版本的save()方法。

    DomesticCat pk = new DomesticCat(); pk.setColor(Color.TABBY); pk.setSex('F'); pk.setName("PK"); pk.setKittens( new HashSet() ); pk.addKitten(fritz); sess.save( pk, new Long(1234) );

    如果你持久化的對象有關聯的對象(associated objects)(例如上例中的kittens集合) 那么對這些對象(譯注:pk和kittens)進行持久化的順序是任意的(也就是說可以先對kittens進行持久化也可以先對pk進行持久化), 除非你在外鍵列上有NOT NULL約束。 Hibernate不會違反外鍵約束,但是如果你用錯誤的順序持久化對象(譯注:在pk持久之前持久kitten),那么可能會違反NOT NULL約束。

    通常你不會為這些細節煩心,因為你很可能會使用Hibernate的 傳播性持久化(transitive persistence)功能自動保存相關聯那些對象。 這樣連違反NOT NULL約束情況都不會出現了 - Hibernate會管好所有的事情。 傳播性持久化(transitive persistence)將在本章稍后討論。

    10.3.?裝載對象

    如果你知道某個實例的持久化標識(identifier),你就可以使用Session的load()方法 來獲取它。 load()的另一個參數是指定類的.class對象。 本方法會創建指定類的持久化實例,并從數據庫加載其數據(state)。

    Cat fritz = (Cat) sess.load(Cat.class, generatedId); // you need to wrap primitive identifiers long pkId = 1234; DomesticCat pk = (DomesticCat) sess.load( Cat.class, new Long(pkId) );

    此外, 你可以把數據(state)加載到指定的對象實例上(覆蓋掉該實例原來的數據)。

    Cat cat = new DomesticCat(); // load pk's state into cat sess.load( cat, new Long(pkId) ); Set kittens = cat.getKittens();

    請注意如果沒有匹配的數據庫記錄,load()方法可能拋出無法恢復的異常(unrecoverable exception)。 如果類的映射使用了代理(proxy),load()方法會返回一個未初始化的代理,直到你調用該代理的某方法時才會去訪問數據庫。 若你希望在某對象中創建一個指向另一個對象的關聯,又不想在從數據庫中裝載該對象時同時裝載相關聯的那個對象,那么這種操作方式就用得上的了。 如果為相應類映射關系設置了batch-size, 那么使用這種操作方式允許多個對象被一批裝載(因為返回的是代理,無需從數據庫中抓取所有對象的數據)。

    如果你不確定是否有匹配的行存在,應該使用get()方法,它會立刻訪問數據庫,如果沒有對應的行,會返回null。

    Cat cat = (Cat) sess.get(Cat.class, id); if (cat==null) {cat = new Cat();sess.save(cat, id); } return cat;

    你甚至可以選用某個LockMode,用SQL的SELECT ... FOR UPDATE裝載對象。 請查閱API文檔以獲取更多信息。

    Cat cat = (Cat) sess.get(Cat.class, id, LockMode.UPGRADE);

    注意,任何關聯的對象或者包含的集合都不會被以FOR UPDATE方式返回, 除非你指定了lock或者all作為關聯(association)的級聯風格(cascade style)。

    任何時候都可以使用refresh()方法強迫裝載對象和它的集合。如果你使用數據庫觸發器功能來處理對象的某些屬性,這個方法就很有用了。

    sess.save(cat); sess.flush(); //force the SQL INSERT sess.refresh(cat); //re-read the state (after the trigger executes)

    此處通常會出現一個重要問題: Hibernate會從數據庫中裝載多少東西?會執行多少條相應的SQLSELECT語句? 這取決于抓取策略(fetching strategy),會在第?19.1?節 “ 抓取策略(Fetching strategies) ”中解釋。

    10.4.?查詢

    如果不知道所要尋找的對象的持久化標識,那么你需要使用查詢。Hibernate支持強大且易于使用的面向對象查詢語言(HQL)。 如果希望通過編程的方式創建查詢,Hibernate提供了完善的按條件(Query By Criteria, QBC)以及按樣例(Query By Example, QBE)進行查詢的功能。 你也可以用原生SQL(native SQL)描述查詢,Hibernate提供了將結果集(result set)轉化為對象的部分支持。

    10.4.1.?執行查詢

    HQL和原生SQL(native SQL)查詢要通過為org.hibernate.Query的實例來表達。 這個接口提供了參數綁定、結果集處理以及運行實際查詢的方法。 你總是可以通過當前Session獲取一個Query對象:

    List cats = session.createQuery("from Cat as cat where cat.birthdate < ?").setDate(0, date).list();List mothers = session.createQuery("select mother from Cat as cat join cat.mother as mother where cat.name = ?").setString(0, name).list();List kittens = session.createQuery("from Cat as cat where cat.mother = ?").setEntity(0, pk).list();Cat mother = (Cat) session.createQuery("select cat.mother from Cat as cat where cat = ?").setEntity(0, izi).uniqueResult();

    一個查詢通常在調用list()時被執行,執行結果會完全裝載進內存中的一個集合(collection)。 查詢返回的對象處于持久(persistent)狀態。如果你知道的查詢只會返回一個對象,可使用list()的快捷方式uniqueResult()。

    10.4.1.1.?迭代式獲取結果(Iterating results)

    某些情況下,你可以使用iterate()方法得到更好的性能。 這通常是你預期返回的結果在session,或二級緩存(second-level cache)中已經存在時的情況。 如若不然,iterate()會比list()慢,而且可能簡單查詢也需要進行多次數據庫訪問: iterate()會首先使用1條語句得到所有對象的持久化標識(identifiers),再根據持久化標識執行n條附加的select語句實例化實際的對象。

    // fetch ids Iterator iter = sess.createQuery("from eg.Qux q order by q.likeliness").iterate(); while ( iter.hasNext() ) {Qux qux = (Qux) iter.next(); // fetch the object// something we couldnt express in the queryif ( qux.calculateComplicatedAlgorithm() ) {// delete the current instanceiter.remove();// dont need to process the restbreak;} }

    10.4.1.2.?返回元組(tuples)的查詢

    (譯注:元組(tuples)指一條結果行包含多個對象) Hibernate查詢有時返回元組(tuples),每個元組(tuples)以數組的形式返回:

    Iterator kittensAndMothers = sess.createQuery("select kitten, mother from Cat kitten join kitten.mother mother").list().iterator();while ( kittensAndMothers.hasNext() ) {Object[] tuple = (Object[]) kittensAndMothers.next();Cat kitten = tuple[0];Cat mother = tuple[1];.... }

    10.4.1.3.?標量(Scalar)結果

    查詢可在select從句中指定類的屬性,甚至可以調用SQL統計(aggregate)函數。 屬性或統計結果被認定為"標量(Scalar)"的結果(而不是持久(persistent state)的實體)。

    Iterator results = sess.createQuery("select cat.color, min(cat.birthdate), count(cat) from Cat cat " +"group by cat.color").list().iterator();while ( results.hasNext() ) {Object[] row = results.next();Color type = (Color) row[0];Date oldest = (Date) row[1];Integer count = (Integer) row[2];..... }

    10.4.1.4.?綁定參數

    接口Query提供了對命名參數(named parameters)、JDBC風格的問號(?)參數進行綁定的方法。 不同于JDBC,Hibernate對參數從0開始計數。 命名參數(named parameters)在查詢字符串中是形如:name的標識符。 命名參數(named parameters)的優點是:

    • 命名參數(named parameters)與其在查詢串中出現的順序無關

    • 它們可在同一查詢串中多次出現

    • 它們本身是自我說明的

    //named parameter (preferred) Query q = sess.createQuery("from DomesticCat cat where cat.name = :name"); q.setString("name", "Fritz"); Iterator cats = q.iterate(); //positional parameter Query q = sess.createQuery("from DomesticCat cat where cat.name = ?"); q.setString(0, "Izi"); Iterator cats = q.iterate(); //named parameter list List names = new ArrayList(); names.add("Izi"); names.add("Fritz"); Query q = sess.createQuery("from DomesticCat cat where cat.name in (:namesList)"); q.setParameterList("namesList", names); List cats = q.list();

    10.4.1.5.?分頁

    如果你需要指定結果集的范圍(希望返回的最大行數/或開始的行數),應該使用Query接口提供的方法:

    Query q = sess.createQuery("from DomesticCat cat"); q.setFirstResult(20); q.setMaxResults(10); List cats = q.list();

    Hibernate 知道如何將這個有限定條件的查詢轉換成你的數據庫的原生SQL(native SQL)。

    10.4.1.6.?可滾動遍歷(Scrollable iteration)

    如果你的JDBC驅動支持可滾動的ResuleSet,Query接口可以使用ScrollableResults,允許你在查詢結果中靈活游走。

    Query q = sess.createQuery("select cat.name, cat from DomesticCat cat " +"order by cat.name"); ScrollableResults cats = q.scroll(); if ( cats.first() ) {// find the first name on each page of an alphabetical list of cats by namefirstNamesOfPages = new ArrayList();do {String name = cats.getString(0);firstNamesOfPages.add(name);}while ( cats.scroll(PAGE_SIZE) );// Now get the first page of catspageOfCats = new ArrayList();cats.beforeFirst();int i=0;while( ( PAGE_SIZE > i++ ) && cats.next() ) pageOfCats.add( cats.get(1) );} cats.close()

    請注意,使用此功能需要保持數據庫連接(以及游標(cursor))處于一直打開狀態。 如果你需要斷開連接使用分頁功能,請使用setMaxResult()/setFirstResult()

    10.4.1.7.?外置命名查詢(Externalizing named queries)

    你可以在映射文件中定義命名查詢(named queries)。 (如果你的查詢串中包含可能被解釋為XML標記(markup)的字符,別忘了用CDATA包裹起來。)

    <query name="eg.DomesticCat.by.name.and.minimum.weight"><![CDATA[from eg.DomesticCat as catwhere cat.name = ?and cat.weight > ? ] ]></query>

    參數綁定及執行以編程方式(programatically)完成:

    Query q = sess.getNamedQuery("eg.DomesticCat.by.name.and.minimum.weight"); q.setString(0, name); q.setInt(1, minWeight); List cats = q.list();

    請注意實際的程序代碼與所用的查詢語言無關,你也可在元數據中定義原生SQL(native SQL)查詢, 或將原有的其他的查詢語句放在配置文件中,這樣就可以讓Hibernate統一管理,達到遷移的目的。

    10.4.2.?過濾集合

    集合過濾器(filter)是一種用于一個持久化集合或者數組的特殊的查詢。查詢字符串中可以使用"this"來引用集合中的當前元素。

    Collection blackKittens = session.createFilter(pk.getKittens(), "where this.color = ?").setParameter( Color.BLACK, Hibernate.custom(ColorUserType.class) ).list() );

    返回的集合可以被認為是一個包(bag, 無順序可重復的集合(collection)),它是所給集合的副本。 原來的集合不會被改動(這與“過濾器(filter)”的隱含的含義不符,不過與我們期待的行為一致)。

    請注意過濾器(filter)并不需要from子句(當然需要的話它們也可以加上)。過濾器(filter)不限定于只能返回集合元素本身。

    Collection blackKittenMates = session.createFilter(pk.getKittens(), "select this.mate where this.color = eg.Color.BLACK.intValue").list();

    即使無條件的過濾器(filter)也是有意義的。例如,用于加載一個大集合的子集:

    Collection tenKittens = session.createFilter(mother.getKittens(), "").setFirstResult(0).setMaxResults(10).list();

    10.4.3.?條件查詢(Criteria queries)

    HQL極為強大,但是有些人希望能夠動態的使用一種面向對象API創建查詢,而非在他們的Java代碼中嵌入字符串。對于那部分人來說,Hibernate提供了直觀的Criteria查詢API。

    Criteria crit = session.createCriteria(Cat.class); crit.add( Expression.eq( "color", eg.Color.BLACK ) ); crit.setMaxResults(10); List cats = crit.list();

    Criteria以及相關的樣例(Example)API將會再第?15?章 條件查詢(Criteria Queries) 中詳細討論。

    10.4.4.?使用原生SQL的查詢

    你可以使用createSQLQuery()方法,用SQL來描述查詢,并由Hibernate處理將結果集轉換成對象的工作。 請注意,你可以在任何時候調用session.connection()來獲得并使用JDBC Connection對象。 如果你選擇使用Hibernate的API, 你必須把SQL別名用大括號包圍起來:

    List cats = session.createSQLQuery("SELECT {cat.*} FROM CAT {cat} WHERE ROWNUM<10","cat",Cat.class ).list(); List cats = session.createSQLQuery("SELECT {cat}.ID AS {cat.id}, {cat}.SEX AS {cat.sex}, " +"{cat}.MATE AS {cat.mate}, {cat}.SUBCLASS AS {cat.class}, ... " +"FROM CAT {cat} WHERE ROWNUM<10","cat",Cat.class ).list()

    和Hibernate查詢一樣,SQL查詢也可以包含命名參數和占位參數。 可以在第?16?章 Native SQL查詢找到更多關于Hibernate中原生SQL(native SQL)的信息。

    10.5.?修改持久對象

    事務中的持久實例(就是通過session裝載、保存、創建或者查詢出的對象) 被應用程序操作所造成的任何修改都會在Session被刷出(flushed)的時候被持久化(本章后面會詳細討論)。 這里不需要調用某個特定的方法(比如update(),設計它的目的是不同的)將你的修改持久化。 所以最直接的更新一個對象的方法就是在Session處于打開狀態時load()它,然后直接修改即可:

    DomesticCat cat = (DomesticCat) sess.load( Cat.class, new Long(69) ); cat.setName("PK"); sess.flush(); // changes to cat are automatically detected and persisted

    有時這種程序模型效率低下,因為它在同一Session里需要一條SQL SELECT語句(用于加載對象) 以及一條SQL UPDATE語句(持久化更新的狀態)。 為此Hibernate提供了另一種途徑,使用脫管(detached)實例。

    請注意Hibernate本身不提供直接執行UPDATE或DELETE語句的API。 Hibernate提供的是狀態管理(state management)服務,你不必考慮要使用的語句(statements)。 JDBC是出色的執行SQL語句的API,任何時候調用session.connection()你都可以得到一個JDBC Connection對象。 此外,在聯機事務處理(OLTP)程序中,大量操作(mass operations)與對象/關系映射的觀點是相沖突的。 Hibernate的將來版本可能會提供專門的進行大量操作(mass operation)的功能。 參考第?13?章 批量處理(Batch processing),尋找一些可用的批量(batch)操作技巧。

    10.6.?修改脫管(Detached)對象

    很多程序需要在某個事務中獲取對象,然后將對象發送到界面層去操作,最后在一個新的事務保存所做的修改。 在高并發訪問的環境中使用這種方式,通常使用附帶版本信息的數據來保證這些“長“工作單元之間的隔離。

    Hibernate通過提供使用Session.update()或Session.merge()方法 重新關聯脫管實例的辦法來支持這種模型。

    // in the first session Cat cat = (Cat) firstSession.load(Cat.class, catId); Cat potentialMate = new Cat(); firstSession.save(potentialMate);// in a higher layer of the application cat.setMate(potentialMate);// later, in a new session secondSession.update(cat); // update cat secondSession.update(mate); // update mate

    如果具有catId持久化標識的Cat之前已經被另一Session(secondSession)裝載了, 應用程序進行重關聯操作(reattach)的時候會拋出一個異常。

    如果你確定當前session沒有包含與之具有相同持久化標識的持久實例,使用update()。 如果想隨時合并你的的改動而不考慮session的狀態,使用merge()。 換句話說,在一個新session中通常第一個調用的是update()方法,以便保證重新關聯脫管(detached)對象的操作首先被執行。

    希望相關聯的脫管對象(通過引用“可到達”的脫管對象)的數據也要更新到數據庫時(并且也僅僅在這種情況), 應用程序需要對該相關聯的脫管對象單獨調用update() 當然這些可以自動完成,即通過使用傳播性持久化(transitive persistence),請看第?10.11?節 “傳播性持久化(transitive persistence)”。

    lock()方法也允許程序重新關聯某個對象到一個新session上。不過,該脫管(detached)的對象必須是沒有修改過的!

    //just reassociate: sess.lock(fritz, LockMode.NONE); //do a version check, then reassociate: sess.lock(izi, LockMode.READ); //do a version check, using SELECT ... FOR UPDATE, then reassociate: sess.lock(pk, LockMode.UPGRADE);

    請注意,lock()可以搭配多種LockMode, 更多信息請閱讀API文檔以及關于事務處理(transaction handling)的章節。重新關聯不是lock()的唯一用途。

    其他用于長時間工作單元的模型會在第?11.3?節 “樂觀并發控制(Optimistic concurrency control)”中討論。

    10.7.?自動狀態檢測

    Hibernate的用戶曾要求一個既可自動分配新持久化標識(identifier)保存瞬時(transient)對象,又可更新/重新關聯脫管(detached)實例的通用方法。 saveOrUpdate()方法實現了這個功能。

    // in the first session Cat cat = (Cat) firstSession.load(Cat.class, catID);// in a higher tier of the application Cat mate = new Cat(); cat.setMate(mate);// later, in a new session secondSession.saveOrUpdate(cat); // update existing state (cat has a non-null id) secondSession.saveOrUpdate(mate); // save the new instance (mate has a null id)

    saveOrUpdate()用途和語義可能會使新用戶感到迷惑。 首先,只要你沒有嘗試在某個session中使用來自另一session的實例,你應該就不需要使用update(), saveOrUpdate(),或merge()。有些程序從來不用這些方法。

    通常下面的場景會使用update()或saveOrUpdate():

    • 程序在第一個session中加載對象

    • 該對象被傳遞到表現層

    • 對象發生了一些改動

    • 該對象被返回到業務邏輯層

    • 程序調用第二個session的update()方法持久這些改動

    saveOrUpdate()做下面的事:

    • 如果對象已經在本session中持久化了,不做任何事

    • 如果另一個與本session關聯的對象擁有相同的持久化標識(identifier),拋出一個異常

    • 如果對象沒有持久化標識(identifier)屬性,對其調用save()

    • 如果對象的持久標識(identifier)表明其是一個新實例化的對象,對其調用save()

    • 如果對象是附帶版本信息的(通過<version>或<timestamp>) 并且版本屬性的值表明其是一個新實例化的對象,save()它。

    • 否則update() 這個對象

    merge()可非常不同:

    • 如果session中存在相同持久化標識(identifier)的實例,用用戶給出的對象的狀態覆蓋舊有的持久實例

    • 如果session沒有相應的持久實例,則嘗試從數據庫中加載,或創建新的持久化實例

    • 最后返回該持久實例

    • 用戶給出的這個對象沒有被關聯到session上,它依舊是脫管的

    10.8.?刪除持久對象

    使用Session.delete()會把對象的狀態從數據庫中移除。 當然,你的應用程序可能仍然持有一個指向已刪除對象的引用。所以,最好這樣理解:delete()的用途是把一個持久實例變成瞬時(transient)實例。

    sess.delete(cat);

    你可以用你喜歡的任何順序刪除對象,不用擔心外鍵約束沖突。當然,如果你搞錯了順序,還是有可能引發在外鍵字段定義的NOT NULL約束沖突。 例如你刪除了父對象,但是忘記刪除孩子們。

    10.9.?在兩個不同數據庫間復制對象

    偶爾會用到不重新生成持久化標識(identifier),將持久實例以及其關聯的實例持久到不同的數據庫中的操作。

    //retrieve a cat from one database Session session1 = factory1.openSession(); Transaction tx1 = session1.beginTransaction(); Cat cat = session1.get(Cat.class, catId); tx1.commit(); session1.close();//reconcile with a second database Session session2 = factory2.openSession(); Transaction tx2 = session2.beginTransaction(); session2.replicate(cat, ReplicationMode.LATEST_VERSION); tx2.commit(); session2.close();

    ReplicationMode決定數據庫中已存在相同行時,replicate()如何處理。

    • ReplicationMode.IGNORE - 忽略它

    • ReplicationMode.OVERWRITE - 覆蓋相同的行

    • ReplicationMode.EXCEPTION - 拋出異常

    • ReplicationMode.LATEST_VERSION - 如果當前的版本較新,則覆蓋,否則忽略

    這個功能的用途包括使錄入的數據在不同數據庫中一致,產品升級時升級系統配置信息,回滾non-ACID事務中的修改等等。 (譯注,non-ACID,非ACID;ACID,Atomic,Consistent,Isolated and Durable的縮寫)

    10.10.?Session刷出(flush)

    每間隔一段時間,Session會執行一些必需的SQL語句來把內存中的對象的狀態同步到JDBC連接中。這個過程被稱為刷出(flush),默認會在下面的時間點執行:

    • 在某些查詢執行之前

    • 在調用org.hibernate.Transaction.commit()的時候

    • 在調用Session.flush()的時候

    涉及的SQL語句會按照下面的順序發出執行:

  • 所有對實體進行插入的語句,其順序按照對象執行Session.save()的時間順序

  • 所有對實體進行更新的語句

  • 所有進行集合刪除的語句

  • 所有對集合元素進行刪除,更新或者插入的語句

  • 所有進行集合插入的語句

  • 所有對實體進行刪除的語句,其順序按照對象執行Session.delete()的時間順序

  • (有一個例外是,如果對象使用native方式來生成ID(持久化標識)的話,它們一執行save就會被插入。)

    除非你明確地發出了flush()指令,關于Session何時會執行這些JDBC調用是完全無法保證的,只能保證它們執行的前后順序。 當然,Hibernate保證,Query.list(..)絕對不會返回已經失效的數據,也不會返回錯誤數據。

    也可以改變默認的設置,來讓刷出(flush)操作發生的不那么頻繁。 FlushMode類定義了三種不同的方式。 僅在提交時刷出(僅當Hibernate的Transaction API被使用時有效), 按照剛才說的方式刷出, 以及除非明確使用flush()否則從不刷出。 最后一種模式對于那些需要長時間保持Session為打開或者斷線狀態的長時間運行的工作單元很有用。 (參見 第?11.3.2?節 “長生命周期session和自動版本化”).

    sess = sf.openSession(); Transaction tx = sess.beginTransaction(); sess.setFlushMode(FlushMode.COMMIT); // allow queries to return stale stateCat izi = (Cat) sess.load(Cat.class, id); izi.setName(iznizi);// might return stale data sess.find("from Cat as cat left outer join cat.kittens kitten");// change to izi is not flushed! ... tx.commit(); // flush occurs

    刷出(flush)期間,可能會拋出異常。(例如一個DML操作違反了約束) 異常處理涉及到對Hibernate事務性行為的理解,因此我們將在第?11?章 事務和并發中討論。

    10.11.?傳播性持久化(transitive persistence)

    對每一個對象都要執行保存,刪除或重關聯操作讓人感覺有點麻煩,尤其是在處理許多彼此關聯的對象的時候。 一個常見的例子是父子關系。考慮下面的例子:

    如果一個父子關系中的子對象是值類型(value typed)(例如,地址或字符串的集合)的,他們的生命周期會依賴于父對象,可以享受方便的級聯操作(Cascading),不需要額外的動作。 父對象被保存時,這些值類型(value typed)子對象也將被保存;父對象被刪除時,子對象也將被刪除。 這對將一個子對象從集合中移除是同樣有效:Hibernate會檢測到,并且因為值類型(value typed)的對象不可能被其他對象引用,所以Hibernate會在數據庫中刪除這個子對象。

    現在考慮同樣的場景,不過父子對象都是實體(entities)類型,而非值類型(value typed)(例如,類別與個體,或母貓和小貓)。 實體有自己的生命期,允許共享對其的引用(因此從集合中移除一個實體,不意味著它可以被刪除), 并且實體到其他關聯實體之間默認沒有級聯操作的設置。 Hibernate默認不實現所謂的可到達即持久化(persistence by reachability)的策略。

    每個Hibernate session的基本操作 - 包括 persist(), merge(), saveOrUpdate(), delete(), lock(), refresh(), evict(), replicate() - 都有對應的級聯風格(cascade style)。 這些級聯風格(cascade style)風格分別命名為 create, merge, save-update, delete, lock, refresh, evict, replicate。 如果你希望一個操作被順著關聯關系級聯傳播,你必須在映射文件中指出這一點。例如:

    <one-to-one name="person" cascade="create"/>

    級聯風格(cascade style)是可組合的:

    <one-to-one name="person" cascade="create,delete,lock"/>

    你可以使用cascade="all"來指定全部操作都順著關聯關系級聯(cascaded)。 默認值是cascade="none",即任何操作都不會被級聯(cascaded)。

    注意有一個特殊的級聯風格(cascade style) delete-orphan,只應用于one-to-many關聯,表明delete()操作 應該被應用于所有從關聯中刪除的對象。

    建議:

    • 通常在<many-to-one>或<many-to-many>關系中應用級聯(cascade)沒什么意義。 級聯(cascade)通常在 <one-to-one>和<one-to-many>關系中比較有用。

    • 如果子對象的壽命限定在父親對象的壽命之內,可通過指定cascade="all,delete-orphan"將其變為自動生命周期管理的對象(lifecycle object)

    • 其他情況,你可根本不需要級聯(cascade)。但是如果你認為你會經常在某個事務中同時用到父對象與子對象,并且你希望少打點兒字,可以考慮使用cascade="create,merge,save-update"。

    可以使用cascade="all"將一個關聯關系(無論是對值對象的關聯,或者對一個集合的關聯)標記為父/子關系的關聯。 這樣對父對象進行save/update/delete操作就會導致子對象也進行save/update/delete操作。

    此外,一個持久的父對象對子對象的淺引用(mere reference)會導致子對象被同步save/update。 不過,這個隱喻(metaphor)的說法并不完整。除非關聯是<one-to-many>關聯并且被標記為cascade="delete-orphan", 否則父對象失去對某個子對象的引用不會導致該子對象被自動刪除。 父子關系的級聯(cascading)操作準確語義如下:

    • 如果父對象被persist(),那么所有子對象也會被persist()

    • 如果父對象被merge(),那么所有子對象也會被merge()

    • 如果父對象被save(),update()或 saveOrUpdate(),那么所有子對象則會被saveOrUpdate()

    • 如果某個持久的父對象引用了瞬時(transient)或者脫管(detached)的子對象,那么子對象將會被saveOrUpdate()

    • 如果父對象被刪除,那么所有子對象也會被delete()

    • 除非被標記為cascade="delete-orphan"(刪除“孤兒”模式,此時不被任何一個父對象引用的子對象會被刪除), 否則子對象失掉父對象對其的引用時,什么事也不會發生。 如果有特殊需要,應用程序可通過顯式調用delete()刪除子對象。

    10.12.?使用元數據

    Hibernate中有一個非常豐富的元級別(meta-level)的模型,含有所有的實體和值類型數據的元數據。 有時這個模型對應用程序本身也會非常有用。 比如說,應用程序可能在實現一種“智能”的深度拷貝算法時, 通過使用Hibernate的元數據來了解哪些對象應該被拷貝(比如,可變的值類型數據), 那些不應該(不可變的值類型數據,也許還有某些被關聯的實體)。

    Hibernate提供了ClassMetadata接口,CollectionMetadata接口和Type層次體系來訪問元數據。 可以通過SessionFactory獲取元數據接口的實例。

    Cat fritz = ......; ClassMetadata catMeta = sessionfactory.getClassMetadata(Cat.class);Object[] propertyValues = catMeta.getPropertyValues(fritz); String[] propertyNames = catMeta.getPropertyNames(); Type[] propertyTypes = catMeta.getPropertyTypes();// get a Map of all properties which are not collections or associations Map namedValues = new HashMap(); for ( int i=0; i<propertyNames.length; i++ ) {if ( !propertyTypes[i].isEntityType() && !propertyTypes[i].isCollectionType() ) {namedValues.put( propertyNames[i], propertyValues[i] );} }

    ?

    總結

    以上是生活随笔為你收集整理的session 中对象实例在不同事务中的状态的全部內容,希望文章能夠幫你解決所遇到的問題。

    如果覺得生活随笔網站內容還不錯,歡迎將生活随笔推薦給好友。

    国产精品毛片久久蜜 | 国产福利一区二区三区在线观看 | 亚洲黄色免费在线 | 欧美日韩国产一区二 | 成人久久久久久久久 | 在线观看视频一区二区三区 | 国产黄在线 | 国产精品婷婷午夜在线观看 | 男女精品久久 | 高清国产午夜精品久久久久久 | 色天堂在线视频 | 日本中文乱码卡一卡二新区 | 在线 日韩 av | 国内免费的中文字幕 | 在线视频欧美精品 | 一本—道久久a久久精品蜜桃 | av福利在线导航 | 日日天天av| 伊人www22综合色 | 国产精品美女久久久久久久 | 色婷婷www | 天天插天天 | 精品专区一区二区 | 国产精品久久久久久久久大全 | 91在线国内视频 | 欧洲性视频 | 国产成人精品女人久久久 | 91人人爽人人爽人人精88v | 欧美美女视频在线观看 | 色综合狠狠干 | 精品一区二区日韩 | 一本—道久久a久久精品蜜桃 | 四虎国产精品成人免费影视 | 中文字幕精 | 国产免费不卡 | 国产小视频免费观看 | 亚洲国产精品电影 | 国产福利av在线 | 不卡的av电影 | 亚洲精品五月天 | 欧美在线视频一区二区三区 | www.夜夜骑.com | 91热视频 | 久久一级片 | 97碰在线 | 狠狠亚洲| 午夜精品久久久久久久99无限制 | 一区二区三区日韩视频在线观看 | 日本精品在线看 | 91av福利视频 | 久久国产精品99久久久久 | 欧美三人交 | 成人久久久精品国产乱码一区二区 | 中文字幕一区二区三区四区久久 | 国产中文在线字幕 | 国产精品岛国久久久久久久久红粉 | 久草网站在线 | 色综合天天狠天天透天天伊人 | 黄色一级大片在线免费看国产一 | 成人av电影免费在线播放 | 亚洲国产影院av久久久久 | 日本不卡一区二区 | 国产一区网 | 狂野欧美激情性xxxx | 亚洲精品国产日韩 | 国产精品成久久久久 | 国内精品免费 | 亚洲国产97在线精品一区 | 久久久久激情电影 | 在线成人欧美 | 欧美性高跟鞋xxxxhd | 久久国产精品免费观看 | 久久午夜影视 | 夜夜狠狠| 久久免费视频8 | 激情五月伊人 | 91成人免费看片 | 婷婷国产视频 | 国产在线视频一区二区 | 久久久国产精品视频 | 永久中文字幕 | 天天干天天天 | 日韩一级黄色片 | 免费在线视频一区二区 | 在线91精品 | 天海翼一区二区三区免费 | 在线免费av电影 | 成人一级免费视频 | 97精品伊人 | 亚洲天天做 | 久久精品久久综合 | 日韩一区在线播放 | 国产亚洲精品久久久久久久久久 | 波多野结衣在线观看视频 | 欧美三级免费 | 国产成人一区二区三区电影 | 国产视频在线免费 | 国产区精品视频 | 在线观看亚洲精品 | 国产精品福利无圣光在线一区 | 国产91精品看黄网站在线观看动漫 | 国产精品99久久久精品免费观看 | 97操操操| 蜜臀久久99精品久久久酒店新书 | 久久理论电影网 | 国产一卡二卡四卡国 | 国产一级二级在线观看 | 1024在线看片| 国产精品免费在线播放 | 成人一区二区三区中文字幕 | 日韩电影久久 | 亚洲另类视频 | 日韩精品久久久免费观看夜色 | 国产精品一区二区三区在线播放 | 日韩在线一区二区免费 | 亚洲理论片在线观看 | 亚洲成a人片在线观看网站口工 | 亚洲激情综合 | av中文电影 | 91视频成人免费 | 狠狠干在线 | 特级a毛片 | 亚洲一区天堂 | 久久久久久久久亚洲精品 | 久久精品一区二区三区国产主播 | 日韩久久片 | 超碰99在线 | 免费av的网站 | 在线观看韩国av | 久久与婷婷| 日韩视频三区 | 欧美福利片在线观看 | 午夜精品婷婷 | 国产伦理精品一区二区 | 天天艹天天 | 韩国av免费 | 久久亚洲专区 | 久久久久久久久久久黄色 | 精品国产资源 | 美女国产网站 | 国产999精品久久久 免费a网站 | 亚洲视频一级 | 九九视频网站 | 久久a免费视频 | 狠狠操狠狠干天天操 | 高清免费在线视频 | 日韩色av色资源 | 在线视频成人 | 黄色大片免费播放 | 在线视频app | 亚洲欧美视频 | 国产精品v欧美精品v日韩 | 欧美激情视频三区 | 中文字幕制服丝袜av久久 | 国产日韩欧美网站 | 国产在线 一区二区三区 | 亚洲欧美视屏 | www日韩精品 | 国产一区二区免费 | 久久精品www人人爽人人 | 99久久精品免费看国产 | 亚洲综合狠狠干 | 97超碰人人模人人人爽人人爱 | 在线精品国产 | www色网站 | 最新婷婷色| 日本爱爱片| 国产精品久久久毛片 | 国产色婷婷精品综合在线手机播放 | 黄色小视频在线观看免费 | 日韩美一区二区三区 | 久久免费在线观看 | 久久综合狠狠综合久久狠狠色综合 | 免费在线h| 久久久久久国产精品亚洲78 | a视频在线播放 | 亚洲精品在线看 | 国产精品爽爽爽 | 91国内产香蕉 | 麻花豆传媒一二三产区 | 91爱看片 | 最新不卡av| 麻豆久久一区二区 | 欧美精品一区二区在线观看 | 视频国产 | www天天干com | 国产日产精品久久久久快鸭 | 免费在线观看一区二区三区 | 国产精品mv | 在线观看av网站 | av在线播放中文字幕 | 日韩xxxbbb | 久久国产福利 | 在线观看91av | 亚洲欧美va | 中国老女人日b | 日韩簧片在线观看 | 日韩区欠美精品av视频 | 久久伊99综合婷婷久久伊 | 国产精品自产拍在线观看 | 日韩av电影免费观看 | 激情校园亚洲 | 69av视频在线 | 黄色美女免费网站 | www中文在线 | 成人一区二区在线观看 | 色综合久久悠悠 | 最近中文字幕视频网 | 狠狠干天天操 | 9i看片成人免费看片 | 在线色视频小说 | 91精品一区二区三区蜜臀 | 国产精品久久久久久久久久免费 | 97色资源 | 亚洲精品国产区 | 伊人宗合网 | 国产免费区 | 天天操操 | 激情综合网天天干 | 成人a在线观看高清电影 | 午夜在线资源 | 五月婷婷在线观看 | 中文字幕在线观看一区二区三区 | 成人黄色影片在线 | 婷婷久草 | 日b黄色片 | 久久高清av | 国产一二区精品 | 日韩v在线 | 亚洲九九九 | 99久久国产免费看 | 久久久久久久久综合 | 日批在线看 | 人人爽人人爽人人片av免 | 天无日天天操天天干 | 97超碰中文 | 色片网站在线观看 | 成人av在线影院 | 精品欧美乱码久久久久久 | 精品国产乱码久久久久久1区二区 | 黄av资源 | 久久久网址 | 97在线资源| 国产不卡在线视频 | 九九久久久久99精品 | 91视频国产高清 | 中文字幕电影高清在线观看 | 久久久综合电影 | 中文字幕在线观看第三页 | 欧美一级电影在线观看 | 免费日韩一区二区 | 99久久日韩精品视频免费在线观看 | 五月的婷婷 | 日韩经典一区二区三区 | 亚洲精品欧美视频 | 黄色免费国产 | 久久1电影院 | 亚洲精品xxxx | 久草在线在线视频 | 久久夜色网 | 天天综合91| 日日日干| 国产高清一 | 伊人www22综合色 | 99久久婷婷国产精品综合 | 亚洲精品一区二区18漫画 | 免费观看成年人视频 | 免费日韩三级 | 色福利网站 | 国产亚洲精品美女 | 国产九九精品视频 | 夜夜操天天 | www.久久久| 二区三区精品 | 99久久99久国产黄毛片 | 亚洲成a人片77777kkkk1在线观看 | 91久久爱热色涩涩 | 精品国产一区二区三区四 | 日韩在线观看一区二区 | 免费日韩一区二区三区 | 四虎影视成人永久免费观看视频 | 成人av免费在线看 | 国产精品白虎 | 在线视频 国产 日韩 | 亚洲精品久久久久久中文传媒 | 中文字幕在线观看的网站 | 九九在线视频 | 摸bbb搡bbb搡bbbb| 成人在线免费看视频 | 国产精品久久久777 成人手机在线视频 | 精品国产观看 | 亚洲国产视频在线 | 毛片一区二区 | 国产精品女同一区二区三区久久夜 | 亚洲综合色激情五月 | 国产麻豆精品免费视频 | 日韩欧美在线视频一区二区 | 国产一区二区视频在线播放 | 一区二区三区国产欧美 | 久久久久久国产精品 | 麻豆久久一区 | 日韩亚洲欧美中文字幕 | 国产精品视频在线看 | 久久99国产精品自在自在app | 麻豆视频在线观看 | 欧美va在线观看 | 五月天丁香视频 | 麻花豆传媒mv在线观看 | 国产精品视频999 | 九九色在线 | 免费情趣视频 | 久久久国产电影 | 亚洲毛片久久 | 伊人婷婷激情 | 国产午夜视频在线观看 | 色综合久久久网 | 国产理伦在线 | 日本天天色 | а天堂中文最新一区二区三区 | 亚洲最大激情中文字幕 | 久久综合久久综合这里只有精品 | 久久免费视频网站 | 欧美精品在线观看 | 日本黄色免费在线 | 欧日韩在线视频 | 国产大陆亚洲精品国产 | 在线看国产一区 | 99r在线观看| 天天天天天天操 | 日韩二区在线 | 亚洲日本国产精品 | 免费成人在线电影 | 网站你懂的 | 国产a级片免费观看 | 91精品国产综合久久婷婷香蕉 | 日韩av片无码一区二区不卡电影 | 国产高清视频在线播放一区 | 91香蕉视频 mp4 | 91久久精品一区二区二区 | 国产美女久久久 | 99精品视频一区二区 | 欧美日韩不卡在线 | 天天爽人人爽夜夜爽 | 草久久影院 | 日韩欧美视频二区 | 久久久久国产一区二区三区 | 五月婷网站 | 天天干天天色2020 | 日韩激情网 | 久久久久久麻豆 | 丁五月婷婷 | 丁香综合av| 中文字幕电影高清在线观看 | 久久国产福利 | 久久精品人人做人人综合老师 | 久久精品专区 | 人人看人人爱 | 精品国产电影一区 | 午夜av剧场 | 日韩视频中文字幕在线观看 | 色黄久久久久久 | 91日韩免费 | 中文字幕av网站 | 国产精品嫩草69影院 | 菠萝菠萝蜜在线播放 | 日韩狠狠操 | 色综合天天综合 | 成人免费看电影 | 深爱婷婷网 | 美女视频黄,久久 | 亚洲 欧美 综合 在线 精品 | 国内精品视频免费 | 91成人免费在线视频 | 免费看的国产视频网站 | 国产97碰免费视频 | 国产系列在线观看 | 91免费试看 | 国产精品乱码久久久久 | 西西www4444大胆视频 | 日韩综合一区二区 | 久久好看免费视频 | 国产毛片aaa | 国产高清在线精品 | 婷婷激情五月综合 | 中文字幕在线网址 | 日本中文字幕一二区观 | 亚洲欧美国产精品久久久久 | 97色综合| 韩国av免费看 | 精品国产美女 | 国产精品久久久久久久久久久免费看 | av网站在线观看免费 | 国产最新在线 | 欧美一级免费 | 久久一级电影 | 亚洲国产精品久久久 | 91精品久久久久久久久久入口 | 亚洲成成品网站 | 久久dvd | 日韩av有码在线 | 精品国产一区二区三区久久 | 婷婷新五月 | 日韩高清免费在线观看 | 天天性天天草 | 成人黄色电影在线播放 | 久久久国产精品人人片99精片欧美一 | 日韩精品一区在线播放 | 亚洲高清资源 | 综合色站 | 中文字幕在线观看免费高清电影 | 国产精品视频最多的网站 | 久久福利小视频 | 国产中文字幕大全 | 日韩av免费一区二区 | 色婷av| 国产99re| 激情在线网站 | 一级黄色av| 免费网站黄 | 国产最顶级的黄色片在线免费观看 | 久久综合九色99 | 国产又粗又猛又黄又爽的视频 | 亚洲黄色激情小说 | 丁香婷婷成人 | 操操操操网 | 天天射天天做 | 337p日本大胆噜噜噜噜 | 久久久受www免费人成 | 婷婷丁香色综合狠狠色 | 999视频在线播放 | 综合中文字幕 | 深夜免费福利视频 | 天天躁日日躁狠狠躁av麻豆 | 天天草天天摸 | 伊人国产在线观看 | 国产精品 中文字幕 亚洲 欧美 | 久久夜夜夜 | 国产日韩视频在线 | 伊人干综合| 婷婷国产在线观看 | 精品视频免费观看 | 久久免费av | 久久久久久国产一区二区三区 | 国产午夜精品一区二区三区 | 国产无套精品久久久久久 | 最近免费中文字幕大全高清10 | 最近日本韩国中文字幕 | 亚洲韩国一区二区三区 | 中文字幕一区二区三区四区 | 天天综合网天天 | 九九热在线免费观看 | 最新高清无码专区 | 日韩av不卡在线 | 国产裸体永久免费视频网站 | 久久九九网站 | 三级av网站 | 午夜久久久精品 | 97免费视频在线播放 | 国内精品久久久久久久久久久久 | 久久精品99国产精品亚洲最刺激 | 久草com| 香蕉影视在线观看 | 97超碰免费在线 | 久久免费视频6 | 高清有码中文字幕 | 日女人电影 | 亚洲一区二区三区91 | 中文av免费 | 成人动态视频 | 四虎在线视频免费观看 | 成人黄色中文字幕 | 国产精品黑丝在线观看 | 精品国产亚洲日本 | 欧美成人精品欧美一级乱 | 91精彩视频在线观看 | 中文字幕在线观看的网站 | av黄免费看| 亚州激情视频 | 日韩精品一区电影 | 国产一区二区视频在线播放 | 伊人影院av | 日韩精品一区二区三区不卡 | 亚洲一区二区三区在线看 | zzijzzij亚洲成熟少妇 | 色婷婷综合久久久中文字幕 | 久久免费视频网 | 女人18毛片a级毛片一区二区 | 久久艹久久 | 在线播放一区 | 狠狠色丁香久久综合网 | 欧美日韩大片在线观看 | 国产美女视频 | 在线视频精品 | 国产麻豆剧果冻传媒视频播放量 | 五月婷婷中文 | 丁香久久 | 在线电影a | 日韩午夜av| 日韩国产欧美在线播放 | 国产免费不卡 | 亚洲国产精品传媒在线观看 | 色多视频在线观看 | 久久国产精品一国产精品 | 欧美精品一区在线发布 | 特片网久久| 最近日本mv字幕免费观看 | 日日草夜夜操 | 91伊人影院 | 91免费网站在线观看 | 成人影片在线免费观看 | 欧美成亚洲 | 在线视频1卡二卡三卡 | 五月天婷婷在线观看视频 | 亚洲精品18日本一区app | 欧美午夜精品久久久久久孕妇 | 久久国产女人 | 国产天天综合 | 日韩中文字幕免费视频 | 在线天堂视频 | 婷婷六月天综合 | 一区二区 久久 | 又黄又刺激又爽的视频 | 久久人人射| 亚洲狠狠 | 亚洲三级av | 国产精品成人aaaaa网站 | 狠狠色丁香婷婷 | 国产日本高清 | 日本久久久久 | 91成人午夜 | 在线岛国av| 色婷婷在线视频 | 青春草视频在线播放 | 成人h视频在线 | 日韩素人在线观看 | 能在线观看的日韩av | 丁香5月婷婷久久 | 成人国产精品免费 | 日日爽| 久久精品国产一区二区电影 | 国产精品久久99综合免费观看尤物 | 中文字幕人成人 | 一本一道久久a久久精品蜜桃 | 国产精品久久久久婷婷 | 国产精品视频永久免费播放 | 开心丁香婷婷深爱五月 | 日韩精品综合在线 | 国产精品18久久久久久久久 | 在线国产91 | 欧美精品在线一区二区 | 天天性天天草 | 国产精品美女久久久网av | 香蕉久久久久久av成人 | 美女中文字幕 | 国产一级片毛片 | 久久有精品 | 久久久久久高潮国产精品视 | 在线观看国产成人av片 | 国产va精品免费观看 | 婷婷狠狠操 | 欧美色图88 | 亚洲高清视频在线播放 | 久久撸在线视频 | 国产一区视频免费在线观看 | 在线观看一区视频 | 国产美女网 | 丁香花在线观看免费完整版视频 | 日韩精品不卡在线观看 | 日本久久高清视频 | 日韩精品一区二区三区视频播放 | 最近中文字幕免费观看 | 丁香久久婷婷 | 久久免费国产 | 国产分类视频 | 婷婷综合 | 日韩区视频 | 日日夜夜操av | 香蕉97视频观看在线观看 | 黄色网免费 | 国产在线a | 欧美91精品久久久久国产性生爱 | 成人免费在线播放 | 日韩高清激情 | 2023亚洲精品国偷拍自产在线 | 精品亚洲国产视频 | 就要干b| 亚洲精品玖玖玖av在线看 | 亚洲视频大全 | 亚洲美女精品区人人人人 | 激情久久婷婷 | 18网站在线观看 | 在线免费观看麻豆 | 日韩在线观看一区 | 激情网婷婷 | 中文字幕在线网 | 亚洲资源网 | 久久久久久视频 | 日日夜av| 亚洲精品午夜aaa久久久 | 色婷婷综合久久久 | 天天鲁天天干天天射 | 日韩免费不卡av | 免费看的黄色小视频 | www.久久99 | 奇米网444| 亚洲精品国产精品国自产在线 | 久久久久黄色 | 欧美激情视频一区二区三区 | 色橹橹欧美在线观看视频高清 | 三级a毛片 | 欧美色久 | 国产高清视频在线播放一区 | 国产高清视频免费观看 | 2019精品手机国产品在线 | 国产精品一区二区美女视频免费看 | 黄色三级久久 | 婷婷av电影 | 少妇bbb搡bbbb搡bbbb′ | 丁香六月婷 | 日本乱视频 | 伊人夜夜| 国产资源精品 | 日韩成人高清在线 | 久久久久女人精品毛片九一 | 日日摸日日爽 | 中文字幕一区在线观看视频 | 中文av在线天堂 | 91理论片午午伦夜理片久久 | 国产一级免费播放 | 欧美日韩在线精品一区二区 | 亚洲狠狠丁香婷婷综合久久久 | 五月婷综合 | 91精品人成在线观看 | 精品在线一区二区三区 | 91精品啪在线观看国产81旧版 | 最近中文字幕第一页 | 国产91区 | 亚洲精品小视频 | www.av中文字幕.com | 日韩欧美亚州 | 丝袜美腿亚洲 | 日韩理论在线观看 | 日韩电影中文,亚洲精品乱码 | 免费aa大片 | 91在线免费观看国产 | 四虎国产精品免费 | www.国产毛片 | 日韩在线观看一区 | 欧美精品久久久久性色 | 日韩免费一级a毛片在线播放一级 | 久久久91精品国产一区二区精品 | 热久久电影 | 中文字幕高清 | 日韩精品第一区 | 午夜美女福利直播 | 六月婷色 | h动漫中文字幕 | 久久成人一区二区 | 韩日av在线 | 波多野结衣一区 | 国产亚州精品视频 | 超碰人人草 | 午夜久久影院 | 国产精品免费久久久久影院仙踪林 | 久久桃花网 | 日韩av在线小说 | 91热爆在线观看 | 五月天亚洲精品 | 九九免费在线看完整版 | 激情综合五月婷婷 | 国产精品自在线 | 久久黄色免费观看 | 免费h精品视频在线播放 | 午夜av影院| 午夜精品99久久免费 | 99在线免费观看视频 | 欧美激情片在线观看 | 国产在线观看国语版免费 | 中文字幕一区二区三区在线视频 | av在线成人 | 久久美女高清视频 | 色婷婷婷| 在线影院av | 精品在线99 | 精品一区二区在线观看 | 69人人| 久久伊人色综合 | 超碰在线公开免费 | 成人在线免费观看网站 | 91中文字幕一区 | 国产剧情在线一区 | 国产精品18久久久久久久 | 亚洲精品一区二区三区高潮 | 国色天香第二季 | 久久精品网站免费观看 | 久久免费观看少妇a级毛片 久久久久成人免费 | 国产一区在线免费观看 | 三上悠亚一区二区在线观看 | 久久这里只有精品久久 | 国产一区福利 | 蜜臀久久99精品久久久无需会员 | 国产成人亚洲在线观看 | 婷婷色网视频在线播放 | 国产三级午夜理伦三级 | 国产高清一区二区 | 亚洲91在线| 日本成人中文字幕在线观看 | 久久一精品 | 一区二区电影在线观看 | 日韩免费不卡视频 | 日韩欧美精品一区二区三区经典 | 久99久在线视频 | 天天爱天天操 | 亚洲第一区在线观看 | av一区二区三区在线播放 | 国产一区在线播放 | 91国内在线 | 97视频免费播放 | 午夜精品视频福利 | 在线观看视频日韩 | 日韩爱爱片 | 91免费高清 | 激情五月婷婷网 | av在线一| 91看片淫黄大片一级在线观看 | 久草在线视频在线 | 日韩有码在线观看视频 | 国产成人综合精品 | 欧美精品视 | 亚洲免费av片 | 亚洲天堂毛片 | 精品国产免费看 | 日日夜日日干 | 国产成在线观看免费视频 | 九九热精品视频在线播放 | aaa日本高清在线播放免费观看 | 婷婷五天天在线视频 | 国产黄在线播放 | 亚洲国产中文字幕在线视频综合 | 丝袜足交在线 | 国产成人免费观看久久久 | 波多野结衣最新 | 色综合激情久久 | 最新久久免费视频 | 久久精品免费观看 | 91视频一8mav| 久久福利在线 | 97涩涩视频 | 免费高清男女打扑克视频 | 美女在线免费观看视频 | 色视频成人在线观看免 | 91亚洲欧美 | 精品91久久久久 | 亚洲精品日韩一区二区电影 | 久久久久久久久电影 | 91网免费观看 | 超碰免费在线公开 | 韩国精品一区二区三区六区色诱 | 免费人成网ww44kk44 | 婷婷久久一区 | 岛国精品一区二区 | 99热精品国产 | av福利网址导航 | 国产精品都在这里 | 亚洲欧美日本A∨在线观看 青青河边草观看完整版高清 | 国产精品久久久精品 | 天天干天天射天天插 | 98精品国产自产在线观看 | 久久久久久免费网 | 国产在线观看免 | 最新国产福利 | 午夜av在线电影 | 国产一区视频导航 | 国产91全国探花系列在线播放 | 久久99热这里只有精品 | 中文 一区二区 | 中文字幕乱码亚洲精品一区 | 国产精品久久久久高潮 | 综合网伊人| 国产 日韩 在线 亚洲 字幕 中文 | 欧美 日韩 视频 | 欧美激情亚洲综合 | 久久精品免费看 | 视频成人永久免费视频 | 精品视频专区 | 青草视频在线 | av动图| 久久激情久久 | 麻豆久久精品 | 激情综合网五月激情 | 又色又爽的网站 | 在线观看中文字幕一区二区 | 久久99热这里只有精品 | 国产精品毛片一区二区三区 | 91人人爱| 久久婷婷网 | av网站手机在线观看 | 精品成人久久 | 精精国产xxxx视频在线播放 | 天天综合操 | 久久精品国产精品亚洲精品 | 99综合久久| 亚洲激情视频在线 | 超碰97免费观看 | 九九爱免费视频 | 天海翼一区二区三区免费 | 日韩 | 国产一区高清在线 | 国产成人精品av久久 | 国产亚洲成人精品 | 国产精品一二 | 麻豆国产网站入口 | 亚洲精品日韩在线观看 | 久久电影色| 麻豆一区二区 | 五月婷婷激情 | 成人午夜电影久久影院 | 97精品在线观看 | 日韩理论影院 | 91精品啪在线观看国产线免费 | 在线三级av | 五月天电影免费在线观看一区 | 久久免费视频5 | 午夜三级在线 | 免费韩国av | 超碰在线日韩 | 免费a视频在线观看 | 国产精品一区二区果冻传媒 | 在线国产激情视频 | 久久永久免费 | 日韩欧美国产免费播放 | 久热av | 啪啪资源 | 精品三级av | 一区二区精品视频 | 蜜桃av人人夜夜澡人人爽 | 欧美最爽乱淫视频播放 | 成人免费在线电影 | 成人蜜桃视频 | 国产精品麻豆一区二区三区 | а天堂中文最新一区二区三区 | 色姑娘综合 | av在线成人 | 伊人夜夜 | av国产网站 | 天天操天天弄 | 在线免费观看黄色av | 美女黄网久久 | 五月天狠狠操 | 欧美热久久 | 久久婷亚洲五月一区天天躁 | 日韩欧美一区二区三区视频 | 最近中文字幕免费观看 | 欧美性生交大片免网 | 欧美激情亚洲综合 | 我要看黄色一级片 | 国产精品永久免费观看 | 久久久久国产成人免费精品免费 | 人人干人人草 | 伊人国产在线观看 | 91精品啪 | 成 人 黄 色 视频 免费观看 | 81国产精品久久久久久久久久 | 丁香资源影视免费观看 | 99在线观看免费视频精品观看 | 天天干天天拍天天操 | 久热av | 夜又临在线观看 | 黄色精品一区 | 久久一区二区免费视频 | 三三级黄色片之日韩 | 高潮久久久久久久久 | 国产精品一区二区三区在线播放 | 91九色丨porny丨丰满6 | 美女国内精品自产拍在线播放 | 国产成在线观看免费视频 | 992tv在线成人免费观看 | 香蕉一区 | 日韩不卡高清 | 激情综合啪啪 | 欧美国产日韩一区二区三区 | 狠狠色网 | 青青草在久久免费久久免费 | 91视频com | 日韩在线电影一区二区 | 午夜免费福利视频 | 一级成人在线 | 国产无遮挡又黄又爽在线观看 | 日日干夜夜草 | 日日爽日日操 | 特级毛片在线 | 日韩性xxxx | 天天爱天天草 | 国产一级在线观看 | a视频免费在线观看 | 五月天丁香综合 | www.eeuss影院av撸 | 国产精品成人一区二区三区吃奶 | 亚洲成人动漫在线观看 | 毛片激情永久免费 | 在线a亚洲视频播放在线观看 | 狠狠五月婷婷 | 久草男人天堂 | 久久久在线| 天天摸夜夜添 | 樱空桃av| 六月丁香婷婷久久 | 最新国产在线视频 | 国产精品美乳一区二区免费 | 国产精成人品免费观看 | 97国产一区二区 | 久草视频视频在线播放 | 国产一区精品在线 | 麻豆免费精品视频 | 91爱爱电影 | 九精品| 色在线免费观看 | 在线精品观看国产 | 欧美福利片在线观看 | 8x成人免费视频 | 国产精华国产精品 | 亚洲国产欧洲综合997久久, | 视频在线观看亚洲 | 最新影院| h视频在线看 | 天天干天天天 | 亚州精品天堂中文字幕 | 日本久久久久久科技有限公司 | 亚州精品一二三区 | 成人久久18免费网站 | 国产在线国偷精品产拍免费yy | 一区中文字幕在线观看 | 久久婷婷精品 | 国产精品白浆 | 91片在线观看 | 久久高清毛片 | 天天搞天天干 | 色婷婷电影网 | 国产精品一区在线观看你懂的 | 美女国内精品自产拍在线播放 | 成人av免费在线观看 | 最近最新中文字幕 | 久久国产精品小视频 | 日韩理论电影在线 | 天堂va欧美va亚洲va老司机 | 国产精品一区二 | 欧美韩日在线 | 一区二区三区福利 | 不卡的av在线播放 | 色婷婷99| 精品一区二区免费 | 视频在线观看日韩 | 最新极品jizzhd欧美 | 欧美人体xx| 在线观看中文字幕第一页 | 国产精品色视频 | 久久精品999 | 最近中文字幕免费 | 蜜臀av性久久久久蜜臀aⅴ涩爱 | 国模一区二区三区四区 | 国产一区二区不卡在线 | 天天射射天天 | 精品一区av | 久久99九九99精品 | av大片网站| 欧美在线视频一区二区三区 | 免费在线激情电影 | 精品在线观看一区二区三区 | 色综合色综合久久综合频道88 | 亚州人成在线播放 | 97精品国产aⅴ | 美女一二三区 | 亚洲女同ⅹxx女同tv | 亚洲精品视频在线免费播放 | 97碰碰精品嫩模在线播放 | 国产黄免费看 | 久久久国产精华液 | 欧美激情视频在线观看免费 | 日韩在线中文字幕视频 | 国产一二区视频 | av一级片| 色婷五月| 国产原创在线 | 91av在线视频播放 | 亚洲情感电影大片 | 国产精品男女啪啪 | 中文字幕三区 | 久久国产美女 | 天天干com| 日韩v欧美v日本v亚洲v国产v | 久久国产二区 | 国产免费大片 | 一区二区精品在线 | 国产精品免费视频一区二区 | 国产xxxxx在线观看 | 欧美在线日韩在线 | 97精产国品一二三产区在线 | 日本在线免费看 |