精品欧美一区二区三区在线观看 _久久久久国色av免费观看性色_国产精品久久在线观看_亚洲第一综合网站_91精品又粗又猛又爽_小泽玛利亚一区二区免费_91亚洲精品国偷拍自产在线观看 _久久精品视频在线播放_美女精品久久久_欧美日韩国产成人在线

探索 Spring 事務的奧秘

開發
本文我們對Spring事務和設計理念和底層源碼實現并結合相應案例對此進行了深入的分析,希望對你有幫助。

在當今的軟件開發領域,事務管理是確保數據完整性和一致性的關鍵環節。而 Spring 事務,作為一個強大且廣泛應用的事務管理機制,在構建可靠、高效的應用程序中發揮著至關重要的作用。

當我們踏上對 Spring 事務的探索之旅,就仿佛打開了一扇通往復雜而又精妙世界的大門。在這里,我們將見證事務如何在幕后默默工作,協調各種操作,保障數據的準確性和穩定性。它如同一位無聲的守護者,精心呵護著系統的運行。

無論是應對復雜的業務邏輯還是處理大規模的數據交互,Spring 事務都展現出了其非凡的能力和靈活性。通過深入剖析它的原理、特性和應用場景,我們能夠更好地理解如何充分發揮其優勢,解決實際開發中面臨的諸多挑戰。讓我們一同開啟這場精彩的旅程,去揭開 Spring 事務那神秘的面紗,探尋其中蘊含的無盡智慧與可能。

一、詳解Spring中的事務

1.什么是事務

事務在邏輯上可以認為就是把一組操作看作一個動作。這個動作的內容要么都成功,要么都失敗,這樣才能保證結果的準確性、一致性。

如下代碼所示,如果下面這段代碼兩個插入操作不屬于同一個事務的話,結束時只有張三被插入和李四沒有插入,不符合業務上的準確性。

使用事務進行數據庫增刪改查操作時,必須保證當前使用的數據庫引擎支持事務,以MySQL為例,MySQL默認引擎為innodb,他就是支持事務的。若時myisam則不支持事務,無法實現數據回滾。

public void transaction_exception_nested_nested(){
        User1 user1=new User1();
        user1.setName("張三");
        user1Service.addNested(user1);
        //報錯
   throw new RuntimeException();
   
        User2 user2=new User2();
        user2.setName("李四");
        user2Service.addNested(user2);
       
    }

2.事務的特性ACID簡介

英文翻譯成中文大致是:原子性、隔離性、一致性、持久性。分別代表的含義是:

  • 原子性(Atomicity): 屬于同一個事務的操作要么都成功,要么都失敗,直接回滾,數據庫的數據像是沒有被動過一樣。
  • 一致性(Consistency):在事務開始前和事務結束后,數據庫的完整性沒有被破壞,即操作符合數據庫級聯回滾、預設約束、觸發器要求。
  • 隔離性(Isolation)數據庫允許并發操作,使用準確的隔離性原則才能保證數據一致性,而隔離級別有:讀未提交(Read uncommitted)、讀已提交(read committed)、可重復讀(repeatable read)、串行化(Serializable)。
  • 持久性(Durability):持久化的數據不會丟失,即使系統發生故障。

3.Spring支持的兩種事務管理

有兩種姿勢,分別是手動式事務和注解式事務,前者是手動的,比較少使用,對應的類是TransactionTemplate或者TransactionManager,使用示例如下所示:

@Autowired
private TransactionTemplate transactionTemplate;
public void testTransaction() {

        transactionTemplate.execute(new TransactionCallbackWithoutResult() {
            @Override
            protected void doInTransactionWithoutResult(TransactionStatus transactionStatus) {

                try {

                    // ....  業務代碼
                } catch (Exception e){
                    //回滾
                    transactionStatus.setRollbackOnly();
                }

            }
        });
}

或者下面這樣一段代碼,都是通過都是傳入需要進行事務管理的bean定義,進行手動操作管理:

@Autowired
private PlatformTransactionManager transactionManager;

public void testTransaction() {

  TransactionStatus status = transactionManager.getTransaction(new DefaultTransactionDefinition());
          try {
               // ....  業務代碼
              transactionManager.commit(status);
          } catch (Exception e) {
              transactionManager.rollback(status);
          }
}

而后者就比較常用了,基于注解(底層是通過AOP實現的),使用的示例代碼如下所示:

@Transactional
    public void transaction_exception_nested_nested(){
        User1 user1=new User1();
        user1.setName("張三");
        user1Service.addNested(user1);

        User2 user2=new User2();
        user2.setName("李四");
        user2Service.addNested(user2);
        throw new RuntimeException();
    }

4.Spring事務隔離級別和傳播行為有哪些(筆試常問)

先來說說事務隔離級別:

  • default(默認):PlatfromTransactionManager默認的隔離級別,使用數據庫默認的事務隔離級別,除了default 其它幾個Spring事務隔離級別與JDBC事務隔離級別相對應。
  • read_uncommited(讀未提交):一個事務可以讀取另外一個事務未提交的數據,這可能出現臟讀 而且不可重復度,出現幻像讀等。
  • read_commited(讀已提交):一個事務可以讀取另一個事務已經提交的數據,不可以讀取未提交的數據 可以避免臟讀 但是無法避免不可重復讀和幻像讀。
  • repeatTable_read(可重復讀):一個事務可以讀取另外一個事務已經提交的數據,可以避免臟讀的前提下 ,也可以避免不可重復讀,但是還是無法避免幻像讀。
  • serializable(串行化):這是一個花費較高但是比較可靠的事務隔離級別,可以避免臟讀 幻像讀和不可重復讀(事務被處理為順序執行)

Spring事務傳播屬性:

  • required(默認屬性):Propagation.REQUIRED內外部屬于統一事務,一個回滾全部回滾。(后文會有代碼演示)
  • Mandatory:如果當前存在事務,則支持當前事務,如果不存在事務,則拋出異常
  • Never:以非事務方式執行,如果當前存在事務,則拋出異常
  • Supports:如果當前存在事務,則支持當前事務,.如果不存在事務,以非事務方式執行
  • Not_Supports:以非事務方式執行操作,如果存在事務,則掛起當前事務
  • required_new:在外圍方法開啟事務的情況下Propagation.REQUIRES_NEW修飾的內部方法依然會單獨開啟獨立事務,且與外部方法事務也獨立,內部方法之間、內部方法和外部方法事務均相互獨立,互不干擾。
  • Nested:嵌套,支持當前事務,內層事務的執行失敗不會導致外層事務的回滾,但是外層事務的回滾會影響內層事務導致內層事務隨外層事務一同回滾.

二、事務傳播行為進階知識

1.前置鋪墊,代碼示例編寫

為了更好的解答后續的問題,這里我們給出了一個示例,user1表的service:

@Service
public class User1Service {

    @Resource
    private User1Mapper user1Mapper;

    @Transactional(propagation = Propagation.REQUIRED)
    public void addRequired(User1 user){
        user1Mapper.insert(user);
    }


    @Transactional(propagation = Propagation.REQUIRES_NEW)
    public void addRequiresNew(User1 user){
        user1Mapper.insert(user);
    }



    @Transactional(propagation = Propagation.NESTED)
    public void addNested(User1 user){
        user1Mapper.insert(user);
    }



}

user2表的service,可以看到對于數據庫的操作都在注解上標出不同的傳播行為:

@Service
public class User2Service {

    @Resource
    private User2Mapper user2Mapper;

    @Transactional(propagation = Propagation.REQUIRED)
    public void addRequired(User2 user){
        user2Mapper.insert(user);
    }

    @Transactional(propagation = Propagation.REQUIRED)
    public void addRequiredException(User2 user){
        user2Mapper.insert(user);
        throw new RuntimeException();
    }


    @Transactional(propagation = Propagation.REQUIRES_NEW)
    public void addRequiresNew(User2 user){
        user2Mapper.insert(user);
    }

    @Transactional(propagation = Propagation.REQUIRES_NEW)
    public void addRequiresNewException(User2 user){
        user2Mapper.insert(user);
        throw new RuntimeException();
    }


    @Transactional(propagation = Propagation.NESTED)
    public void addNested(User2 user){
        user2Mapper.insert(user);
    }

    @Transactional(propagation = Propagation.NESTED)
    public void addNestedException(User2 user){
        user2Mapper.insert(user);
        throw new RuntimeException();
    }
}

2. TransactionDefinition.PROPAGATION_REQUIRED

它是Spring的默認傳播行為,說白了發生嵌套在內部的事務會和外部的事務融合,所以外部事務報錯了內部事務也會回滾。

如下面這段代碼,外部的方法沒有加事務,且user1Service、user2Service的方法都是PROPAGATION_REQUIRED這個傳播級別,所以外部報錯不影響兩者的內部提交

/**
     * 彼此都有獨立的事務,外部沒有開事務,所以兩者數據都會入庫
     */
    @GetMapping("/test/add1")
    public void notransaction_exception_required_required() {
        User1 user1 = new User1();
        user1.setName("張三");
        user1Service.addRequired(user1);

        User2 user2 = new User2();
        user2.setName("李四");
        user2Service.addRequired(user2);

        throw new RuntimeException();
    }

然后我們再來看看這樣一段代碼,外部沒有加事務,所以內部兩個事務彼此獨立。可以看到user2Service報錯,所以只有user1Service插入成功:

 @GetMapping("/test/add2")
    public void notransaction_required_required_exception() {
     //插入成功
        User1 user1 = new User1();
        user1.setName("張三");
        user1Service.addRequired(user1);

  //事務是獨立的插入失敗
        User2 user2 = new User2();
        user2.setName("李四");
        user2Service.addRequiredException(user2);

        throw new RuntimeException();
    }

最后在看看這個,外部加了事務,也是REQUIRED,所以內部兩個事務與其融合成為一個事務,當外部方法報錯,兩者插入操作都失敗,數據直接回滾:

/**
     * 外部開啟事務,報錯均回滾
     */
    @GetMapping("/test/add3")
    @Transactional(propagation = Propagation.REQUIRED)
    public void transaction_exception_required_required() {
        User1 user1 = new User1();
        user1.setName("張三");
        user1Service.addRequired(user1);

        User2 user2 = new User2();
        user2.setName("李四");
        user2Service.addRequired(user2);


        throw new RuntimeException();
    }

再來看看一個比較好玩的,內外部都有事務,第2個內部事務報錯,由于三者事務融為一體,所以user2Service的錯誤被外部transaction_required_required_exception方法感知,user1Service插入也是失敗的,所以這個方法兩張表都沒有插入數據

 @GetMapping("/test/add4")
    @Transactional(propagation = Propagation.REQUIRED)
    public void transaction_required_required_exception() {
        User1 user1 = new User1();
        user1.setName("張三");
        user1Service.addRequired(user1);

        User2 user2 = new User2();
        user2.setName("李四");
        //錯誤被外部感知,所以所有user1的插入也被回滾了
        user2Service.addRequiredException(user2);
    }

這個也比較特殊,由于三個事務合為一體,所以即使user2Service報錯不被感知,兩張表的數據也還是沒有插入:

@GetMapping("/test/add5")
    @Transactional
    public void transaction_required_required_exception_try() {
        User1 user1 = new User1();
        user1.setName("張三");
        user1Service.addRequired(user1);

        User2 user2 = new User2();
        user2.setName("李四");
        try {
        //雖然異常被捕獲,但是三個內外部事務融合了,一個報錯就全部插入回滾
            user2Service.addRequiredException(user2);
        } catch (Exception e) {
            System.out.println("方法回滾");
        }
    }

總結:Propagation.REQUIRED內外部屬于統一事務,一個回滾全部回滾,無視try塊代碼的捕獲。

3. TransactionDefinition.PROPAGATION_REQUIRES_NEW

我們還是通過看代碼的方式來講述吧:第一個例子,外部沒有加事務,兩個service彼此事務獨立,外部報錯,但是兩者事務都已提交,所以都插入了:

 @GetMapping("/test/add6")
    public void notransaction_exception_requiresNew_requiresNew() {
        User1 user1 = new User1();
        user1.setName("張三");
        user1Service.addRequiresNew(user1);

        User2 user2 = new User2();
        user2.setName("李四");
        user2Service.addRequiresNew(user2);
        throw new RuntimeException();

    }

外部還是沒有開啟事務,user2Service報錯事務回滾,所以只有user1Service插入了。

@GetMapping("/test/add7")
    public void notransaction_requiresNew_requiresNew_exception() {
        User1 user1 = new User1();
        user1.setName("張三");
        //正常插入
        user1Service.addRequiresNew(user1);

        User2 user2 = new User2();
        user2.setName("李四");
        //保存回滾了 
        user2Service.addRequiresNewException(user2);
    }

來看一個綜合的,外部加了REQUIRED,所以內部第一個事務和外部融合,后兩個事務獨立,在外部報錯的情況下只有addRequired回滾。李四、王五均被插入:

@GetMapping("/test/add8")
    @Transactional(propagation = Propagation.REQUIRED)
    public void transaction_exception_required_requiresNew_requiresNew() {
        User1 user1 = new User1();
        user1.setName("張三");
        //和外部事務融合,外部報錯插入被回滾
        user1Service.addRequired(user1);

        User2 user2 = new User2();
        user2.setName("李四");
        // 事務獨立,不受外部影響,正常插入
        user2Service.addRequiresNew(user2);

        User2 user3 = new User2();
        user3.setName("王五");
        // 事務獨立,不受外部影響,正常插入
        user2Service.addRequiresNew(user3);
        throw new RuntimeException();
    }

外部加了事務,由于王五報錯被外部感知,張三的事務和外部融合,所以張三沒有被插入,這題只有李四被插入了:

 @GetMapping("/test/add9")
    @Transactional(propagation = Propagation.REQUIRED)
    public void transaction_required_requiresNew_requiresNew_exception() {
        User1 user1 = new User1();
        user1.setName("張三");
        //和外部融合
        user1Service.addRequired(user1);

        User2 user2 = new User2();
        user2.setName("李四");
        //事務獨立,正常插入
        user2Service.addRequiresNew(user2);

        User2 user3 = new User2();
        user3.setName("王五");
        //報錯,插入被回滾,外部感知到了錯誤,所以張三的插入也被回滾了
        user2Service.addRequiresNewException(user3);
    }

王五報錯回滾,但是錯誤沒有被外部感知到,張三和外部事務融合,正常插入、李四正常插入。

@GetMapping("/test/add10")
    @Transactional(propagation = Propagation.REQUIRED)
    public void transaction_required_requiresNew_requiresNew_exception_try() {
        User1 user1 = new User1();
        user1.setName("張三");
        //正常插入
        user1Service.addRequired(user1);

        User2 user2 = new User2();
        user2.setName("李四");
        //和外部事務彼此獨立,正常插入
        user2Service.addRequiresNew(user2);
        User2 user3 = new User2();
        user3.setName("王五");
        try {
        // 報錯回滾,但錯誤并沒有被外部感知,所以只有這個事務被回滾
            user2Service.addRequiresNewException(user3);
        } catch (Exception e) {
            System.out.println("回滾");
        }
    }

總結: 在外圍方法開啟事務的情況下Propagation.REQUIRES_NEW修飾的內部方法依然會單獨開啟獨立事務,且與外部方法事務也獨立,內部方法之間、內部方法和外部方法事務均相互獨立,互不干擾。

4. TransactionDefinition.PROPAGATION_NESTED

代碼如下,外部沒有事務,張三、李四彼此獨立一個事務,數據均插入,外部異常不影響成功提交:

 @GetMapping("/test/add11")
    public void notransaction_exception_nested_nested() {
        User1 user1 = new User1();
        user1.setName("張三");
        user1Service.addNested(user1);

        User2 user2 = new User2();
        user2.setName("李四");
        user2Service.addNested(user2);
        throw new RuntimeException();
    }

同理,外部沒有事務,后者報錯不影響前者正常插入:

 @GetMapping("/test/add12")
    public void notransaction_nested_nested_exception() {
        User1 user1 = new User1();
        user1.setName("張三");
        //獨立的事務,不受下方報錯影響
        user1Service.addNested(user1);

        User2 user2 = new User2();
        user2.setName("李四");
        //外部沒有事務,報錯回滾
        user2Service.addNestedException(user2);
    }

外部開啟事務(默認級別),內部事務與其融合,一錯全部回滾:

 @GetMapping("/test/add13")
    @Transactional
    public void transaction_exception_nested_nested(){
    //外部開啟事務,所有nest的事務都與外部事務融合,一個報錯全部回滾
        User1 user1=new User1();
        user1.setName("張三");
        user1Service.addNested(user1);

        User2 user2=new User2();
        user2.setName("李四");
        user2Service.addNested(user2);
        throw new RuntimeException();
    }

同上,外部開啟事務后內部事務與外部融合,異常能被感知后回滾了:

@GetMapping("/test/add14")
    @Transactional
    public void transaction_nested_nested_exception(){
        User1 user1=new User1();
        user1.setName("張三");
        user1Service.addNested(user1);

        User2 user2=new User2();
        user2.setName("李四");
        user2Service.addNestedException(user2);
    }

異常捕獲李四的報錯自己消化了,外部不回滾,這就是transaction_nested和REQUIRED的區別:

 @GetMapping("/test/add15")
    @Transactional
    public void transaction_nested_nested_exception_try(){
        User1 user1=new User1();
        user1.setName("張三");
        user1Service.addNested(user1);

        User2 user2=new User2();
        user2.setName("李四");
        try {
            user2Service.addNestedException(user2);
        } catch (Exception e) {
            System.out.println("方法回滾");
        }
    }

總結:如果當前存在事務,則在嵌套事務內執行。如果當前沒有事務,則進行與PROPAGATION_REQUIRED類似的操作。

5.事務的幾個回滾規則

這個我們從源碼的注釋中就能看出端倪了,如下所示,注釋中已經說明了只有運行時異常或者Error可以觸發回滾,對于檢查型異常是不會回滾。

/**

  * <p>By default, a transaction will be rolling back on {@link RuntimeException}
  * and {@link Error} but not on checked exceptions (business exceptions). See
  * org.springframework.transaction.interceptor.DefaultTransactionAttribute#rollbackOn(Throwable)
  */
 Class<? extends Throwable>[] rollbackFor() default {};

那我如果要自定義一個異常使用怎么辦?如下所示即可,在注解上聲明我們需要回滾的錯誤類型即可。

@Transactional(rollbackFor= MyException.class)

6.詳解@Transactional 注解

@Transactional 作用在不同的地方會有不同的效果,我們最常見的用法就是作用于方法上,如果在方法上加該注解,就會將當前方法中的數據庫操作加入事務中。注意方法必須是public否則事務不會生效。 如果作用于類上,則意味著這個類中所有public的方法都會用到事務。接口同理,但我們不建議這么用。

它的常見參數配置如下:

  • 傳播屬性(propagation):事務的傳播行為,默認值為REQUIRED,可選的值在上面介紹過
  • 隔離級別(isolation):事務的隔離級別,默認值采用DEFAULT,可選的值在上面介紹過
  • 回滾規則(rollbackFor):用于指定能夠觸發事務回滾的異常類型,并且可以指定多個異常類型。
  • 只讀屬性(readOnly):指定事務是否為只讀事務,默認值為 false。
  • 超時時間(timeout):事務的超時時間,默認值為-1(不會超時)。如果超過該時間限制但事務還沒有完成,則自動回滾事務。

7.Spring AOP自調用問題

這個問題,我們不妨舉個例子來說吧,首先我們看看下面這段代碼,很明顯如果我們直接調用add17報錯了事務會回滾,原因很簡單,這個method1加了注解,所以如果我們通過api等工具調用method1時,真正執行這段代碼的對象是結果Spring容器bpp處理過的cglib代理類:

 @Transactional
    @GetMapping("/test/add17")
    public void method1(){
        User2 user2 = new User2();
        user2.setName("李四");
        user2Service.insert(user2);
        System.out.println(1/0);
    }

好了,有了上文的鋪墊,我們再來說說嵌套調用失效問題,代碼如下所示,當我們使用接口調用工具調用時,發現method1執行出錯,李四還是被成功插入了,這是為什么呢? 原因很簡單method1執行者并不是cglib代理對象,下面這段method1,完整的代碼應該是this.method1,

 @GetMapping("/test/add16")
    public void add16() {
        method1();

    }

    @Transactional
    @GetMapping("/test/add17")
    public void method1(){
        User2 user2 = new User2();
        user2.setName("李四");
        user2Service.insert(user2);
        System.out.println(1/0);
    }

這就導致執行調用method1調用者是this,而不是cglib代理的增強類,如下圖所示,正是因為調用者不是代理,導致代理根本不知道method1被調用了,所以事務就失效了

如何解決spring自調用問題呢?

最干脆就是調用時避免嵌套使用就好了,但是有時候應該這個方法要依賴外部的處理邏輯,而外部方法又臭又長改造兩量很大導致無法重構。這時候我們只能想別的辦法。我以前解決的辦法就比較干脆了,既然問題的根源是調用對象錯誤,那我就干脆找出這個對象來調用不就解決了?

所以我們的思路是這樣的,如下代碼所示:

首先的controller中假如應用上下文:

  @Autowired
    private ApplicationContext applicationContext;

用這個上下文去容器中把他撈出來調用method1,問題解決

@GetMapping("/test/add16")
    public void add16() {
        TestController t = (TestController) applicationContext.getBean("testController");
        t.method1();


    }

    @Transactional
    @GetMapping("/test/add17")
    public void method1() {
        User2 user2 = new User2();
        user2.setName("李四");
        user2Service.insert(user2);
        System.out.println(1 / 0);
    }

當然這里還有一種方法,將代理的service類注入,因為spring注入的類都是經過cglib增強的類,所以使用注入的bean也能解決問題,只不過寫法很丑陋而已。

@Autowired
private        TestController t;

@GetMapping("/test/add16")
    public void add16() {

        t.method1();


    }

    @Transactional
    @GetMapping("/test/add17")
    public void method1() {
        User2 user2 = new User2();
        user2.setName("李四");
        user2Service.insert(user2);
        System.out.println(1 / 0);
    }

8.事務場景注意事項

整體大概有以下幾點:

  • 正確的設置@Transactional 的rollbackFor 和propagation 屬性,否則事務可能會回滾失敗;
  • 避免同一個類中調用@Transactional 注解的方法,這樣會導致事務失效
  • @Transactional 注解的方法所在的類必須被Spring 管理,否則不生效;
  • @Transactional 注解只有作用到public 方法上事務才生效;
  • 底層使用的數據庫必須支持事務機制,否則不生效;

小結

自此我們Spring事務和設計理念和底層源碼實現并結合相應案例對此進行了深入的分析,希望對你有幫助。

責任編輯:趙寧寧 來源: 寫代碼的SharkChili
相關推薦

2025-01-22 16:00:00

MySQL數據庫Binlog

2023-09-28 09:03:56

開源搜索分析引擎

2024-02-22 10:36:13

SELECT 語句PostgreSQL數據查詢

2013-07-30 12:29:19

Google App Google技術Engine

2024-09-09 16:30:08

Python編程

2024-06-04 15:56:48

Task?.NET異步編程

2011-04-15 17:43:15

Google App Google

2021-08-09 17:13:39

數值運算

2009-10-21 17:36:36

VB基礎教程

2009-07-17 14:03:34

ibatis DAO事務管理

2011-09-16 14:34:20

2023-10-30 07:36:19

Spring事務傳播機制

2023-12-08 07:52:51

Spring項目開發

2020-08-19 09:45:29

Spring數據庫代碼

2009-06-22 09:01:57

Spring聲明式事務

2013-05-07 11:43:47

2022-04-26 21:49:55

Spring事務數據庫

2011-02-28 13:51:30

Spring事物配置
點贊
收藏

51CTO技術棧公眾號

国产日韩在线播放| 欧美sm极限捆绑bd| 日本不卡在线播放| 精品亚洲永久免费| 国产精品一区二区三区美女| 亚洲一二三区视频在线观看| 成人在线中文字幕| 精品在线免费观看视频| 国产成人精品福利| 日韩欧美第一页| 日韩欧美精品久久| 欧美三级网站在线观看| 羞羞色午夜精品一区二区三区| 欧美高清性hdvideosex| 成年丰满熟妇午夜免费视频 | 亚洲国产欧美精品| 色综合久久久久无码专区| 亚洲av片在线观看| 久久午夜精品一区二区| 自拍偷拍亚洲精品| 一区二区在线免费观看视频| 激情视频网站在线播放色| 国产日韩精品一区二区浪潮av | 日韩免费av一区二区三区| 一级特黄aa大片| 亚洲人成在线影院| 在线色欧美三级视频| 校园春色 亚洲色图| 伊人电影在线观看| 久久久国产精品午夜一区ai换脸| 青青久久av北条麻妃海外网| 国产人与禽zoz0性伦| youjizzjizz亚洲| 一区二区三区在线视频免费观看 | 久久男人av| 欧美日韩在线一区二区| 男女激情免费视频| 欧美三级理伦电影| 91在线播放网址| 51精品国产人成在线观看| 久久久蜜桃一区二区| 欧美三级视频| 日韩最新在线视频| 素人fc2av清纯18岁| 国产成人久久精品一区二区三区| 欧美性猛交视频| 国产欧美综合一区| 黄色av网站在线看| 99精品视频在线免费观看| 亚洲最大激情中文字幕| 中文字字幕在线中文乱码| 国产精品vip| 日韩中文字幕精品视频| 熟女俱乐部一区二区| 午夜精品在线| 在线观看日韩国产| r级无码视频在线观看| www.国产区| yjizz视频网站在线播放| 盗摄精品av一区二区三区| 日本国产一区二区三区| www.av视频| 日韩久久电影| 亚洲美女视频网站| 视频免费在线观看| 中文无码日韩欧| 欧美性xxxxxxxx| 无码人妻h动漫| 日本h片在线观看| 久久久久久久精| 欧美激情视频网址| 极品颜值美女露脸啪啪| 国产精品99视频| 日韩一区二区久久久| 长河落日免费高清观看| 欧美日韩一二| 一区二区av在线| 摸摸摸bbb毛毛毛片| 免费观看成人www动漫视频| 欧美一区二区视频免费观看| xxxx在线免费观看| 91嫩草国产线观看亚洲一区二区 | 在线观看av的网址| 3d玉蒲团在线观看| 一二三区精品福利视频| www.激情网| bbw在线视频| 欧美性猛交xxxx久久久| 日本888xxxx| 韩国精品视频在线观看 | 欧美日韩中出| 精品sm在线观看| 人妻av一区二区| 激情婷婷综合| 欧美激情视频在线观看| 天天操夜夜操视频| 国产精品综合在线视频| 欧美日韩精品中文字幕一区二区| 精品黄色免费中文电影在线播放| 亚洲成在线观看| 亚洲国产日韩欧美在线观看| 97久久精品| 中国china体内裑精亚洲片| 69av视频在线| 日本视频中文字幕一区二区三区| 豆国产97在线| chinese偷拍一区二区三区| 亚洲午夜久久久| 久久黄色片网站| 伊人久久大香线蕉综合网站| 久久亚洲一区二区三区四区五区高 | 欧美伊人久久| 日本中文字幕久久看| 99热精品在线播放| 白白色亚洲国产精品| 亚洲精品一区二区三区蜜桃久| 美女网站视频在线| 欧美日韩国产综合草草| 国产精品边吃奶边做爽| 亚洲天堂免费| 国产精品久久久久久av| 天堂成人在线视频| 亚洲乱码国产乱码精品精的特点 | 在线观看免费高清视频97| 精品无码一区二区三区电影桃花| 精品一区二区精品| 奇米精品在线| 久久男人av资源站| 欧美成人女星排名| 精品欧美一区二区久久久久| 乱一区二区av| 日本一区网站| 免费日韩电影| 日韩国产精品视频| 精品无码黑人又粗又大又长| 国产成人高清在线| 制服诱惑一区| 精品亚洲a∨| 在线亚洲国产精品网| 波多野结衣一区二区三区在线 | 日本视频一区二区三区| 免费精品视频一区| 在线手机中文字幕| 亚洲国产一区自拍| 日韩av免费网址| 国产91高潮流白浆在线麻豆 | 欧美久久一二三四区| 亚洲精品视频网址| 日本不卡一区二区三区| 日韩偷拍一区二区| 国精产品一区一区三区四川| 亚洲色图18p| 一级久久久久久| 国产精品色哟哟网站| 亚洲天堂伊人网| 亚洲字幕久久| 99视频在线播放| 操喷在线视频| 亚洲黄页网在线观看| 丰满少妇乱子伦精品看片| gogo大胆日本视频一区| 精品国产免费av| 免费成人网www| 国产精品高精视频免费| 调教视频免费在线观看| 欧美一区二区三区在线观看视频| 欧美aaa级片| 免费成人小视频| 视频一区二区视频| 中文字幕一区二区三区中文字幕| 国内揄拍国内精品| 欧美69xxxxx| 欧美日韩1区2区| 免费毛片在线播放免费| 91麻豆国产精品久久| 国产xxxxx在线观看| 欧美岛国激情| 国产尤物91| 日日夜夜天天综合| 欧美精品性视频| 香蕉视频黄在线观看| 欧美四级电影网| 91视频免费在线看| 26uuu国产一区二区三区| 一道本视频在线观看| 欧美久久成人| 神马影院一区二区| 亚洲专区**| 国产精品va在线| 日本片在线观看| 亚洲一级一级97网| www久久久com| 91福利精品视频| 久久久精品91| 国产日产欧产精品推荐色| 女人扒开腿免费视频app| 先锋亚洲精品| 蜜臀av性久久久久蜜臀av| 少妇高潮一区二区三区| 91精品久久久久久久久久另类| 日韩av影片| 久久不射热爱视频精品| 久久av少妇| 欧美精品一区二区精品网| 夜夜躁狠狠躁日日躁av| 精品久久久久久久久久| 婷婷久久综合网| 久久久无码精品亚洲日韩按摩| 视频区 图片区 小说区| 天堂av在线一区| 91精品国产91久久久久麻豆 主演| 日韩精品一卡| 欧美日本韩国在线| baoyu135国产精品免费| 成人精品久久av网站| 欧美最新精品| 茄子视频成人在线| av中文在线资源库| 欧美成年人视频网站欧美| a√资源在线| 亚洲免费视频一区二区| 日韩一级片免费观看| 欧美一区二区在线免费播放| 一区二区三区在线免费观看视频| 色欧美88888久久久久久影院| 精品少妇theporn| 亚洲综合一二区| 国产精品 欧美激情| 亚洲欧洲精品一区二区精品久久久| 波多野结衣av在线观看| 99久久精品国产一区| 蜜臀视频在线观看| 国产成人啪免费观看软件| 日韩欧美亚洲另类| 麻豆一区二区三区| 污污视频网站免费观看| 可以看av的网站久久看| 国产成人无码精品久久久性色| 在线欧美视频| 97在线国产视频| 亚洲国产网站| 久久久久久免费看| 一本色道久久综合亚洲精品高清| 亚洲精品无码国产| 夜夜夜久久久| 怡红院av亚洲一区二区三区h| 亚洲国产精品一区| 欧美日韩在线中文| 久久福利影视| 热久久精品国产| 麻豆中文一区二区| 亚洲美女爱爱视频| 国产精品1区2区| 美女网站视频在线观看| 99re这里只有精品6| 免费看黄色的视频| 欧美韩日一区二区三区| 美女网站视频色| 亚洲日本一区二区| 国产这里有精品| 午夜精品久久久久影视| 国产高清中文字幕| 在线视频欧美精品| 国产精品久久免费| 日韩欧美一级片| 天堂av网在线| 在线日韩中文字幕| 亚洲欧美成人影院| 5252色成人免费视频| 天然素人一区二区视频| 成人啪啪免费看| 成人自拍在线| 日韩欧美亚洲区| 亚洲精品久久| 91好吊色国产欧美日韩在线| 日韩影院免费视频| 中文字幕色网站| 99久久99久久精品免费看蜜桃| jizz中文字幕| 一区二区视频免费在线观看| www亚洲视频| 91精品午夜视频| 视频一区二区免费| 中文字幕精品一区久久久久| 天天干在线视频论坛| 日韩av片永久免费网站| 久久久久九九精品影院| 国产一区在线免费观看| 欧美3p视频| 少妇人妻在线视频| 精品一区二区成人精品| 国产三级国产精品| 亚洲欧美日韩中文字幕一区二区三区 | 天堂av免费在线观看| 欧美精品v国产精品v日韩精品 | 日韩精品1区2区| 欧美日韩国产精选| 午夜一区在线观看| 久久人人爽亚洲精品天堂| 色老头在线一区二区三区| 成人综合网网址| 国产一区二区三区四区二区| 欧美视频在线第一页| 日韩国产在线一| 中文字幕第九页| 亚洲人精品一区| 国产成人av免费| 日韩av在线高清| 欧美一卡二卡| 91久久久久久久久久久| 精品国产午夜| 91av资源网| 成人免费看的视频| 国产suv一区二区三区| 欧美性欧美巨大黑白大战| 午夜视频www| 日韩亚洲在线观看| 精品成人av| 欧美日韩视频在线一区二区观看视频| 国产精品magnet| 久久精品一卡二卡| 中文字幕一区二区5566日韩| 免费黄色av片| 日韩黄在线观看| 嗯啊主人调教在线播放视频| 亚洲一区二区自拍| 天天揉久久久久亚洲精品| 免费看污污网站| 国产欧美日韩综合精品一区二区 | 51精品久久久久久久蜜臀| caoporn国产精品免费视频| 青青草原一区二区| 欧美美女在线直播| 国产素人在线观看| 成人av手机在线观看| 国产精品美女毛片真酒店| 日韩欧美的一区二区| 91麻豆免费在线视频| 亚洲a级在线观看| 婷婷伊人综合| 古装做爰无遮挡三级聊斋艳谭| 亚洲免费观看高清完整版在线观看| 综合久久中文字幕| 日韩中文字幕国产精品| 2020国产精品小视频| 91制片厂免费观看| 国产乱人伦偷精品视频免下载| 视频国产一区二区| 91麻豆精品国产91久久久久久 | 91视频国产精品| 国产精品久久久久久| 手机免费看av网站| 综合婷婷亚洲小说| 国产不卡av在线播放| 欧美激情日韩图片| 日韩电影不卡一区| caopor在线视频| 国产精品三级av| 国产伦理一区二区| 欧美激情中文字幕在线| 欧美绝顶高潮抽搐喷水合集| 黄色一级一级片| 国产精品情趣视频| 精品人妻无码一区二区三区蜜桃一| 色综合天天狠天天透天天伊人| 欧美毛片免费观看| 中文久久久久久| 国产精品久久久久久久裸模| 国产乱码精品一区二区三区精东 | 中文字幕精品一区日韩| 国产久卡久卡久卡久卡视频精品| 国产无遮挡免费视频| 亚洲一二在线观看| 久久爱www.| 人妻有码中文字幕| 国产精品传媒入口麻豆| 亚洲精品综合久久| 日本免费久久高清视频| 91超碰国产精品| 国产69视频在线观看| 日本乱人伦aⅴ精品| 最新国产在线拍揄自揄视频| 精品国产乱码久久久久久久软件| 天堂蜜桃91精品| 黄色一级视频免费观看| 亚洲欧美三级伦理| 日本免费一区二区三区视频| 欧美极品欧美精品欧美| 成人免费在线观看入口| 亚洲三区在线播放| 91免费看片网站| 亚洲欧美成人| 国产高潮流白浆| 亚洲午夜av电影| jizz18欧美18| 性生活免费在线观看| 精品久久久久国产| а√天堂8资源在线官网| 欧美亚洲免费高清在线观看| 国产成人在线视频播放|