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

Spring Boot項目業務代碼中使用@Transactional事務失效踩坑點總結

開發 項目管理
Spring 是通過 AOP 技術對方法進行增強實現事務控制的,要調用增強過的方法必然是調用代理后的對象,而這里this是原生對象,并不是代理,自然就沒有事務控制了。

1.概述

接著之前我們對Spring AOP以及基于AOP實現事務控制的上文,今天我們來看看平時在項目業務開發中使用聲明式事務@Transactional的失效場景,并分析其失效原因,從而幫助開發人員盡量避免踩坑。

我們知道 Spring 聲明式事務功能提供了極其方便的事務配置方式,配合 Spring Boot 的自動配置,大多數 Spring Boot 項目只需要在方法上標記 @Transactional 注解,即可一鍵開啟方法的事務性配置。當然后端開發人員對數據庫事務這個概念并不陌生,也知道如果整體考慮多個數據庫操作要么成功要么失敗時,需要通過數據庫事務來實現多個操作的一致性和原子性。如下所示:

@Override
    @Transactional(rollbackFor = Exception.class)
    public void addUser(UserParam param) {
        User user = PtcBeanUtils.copy(param, User.class);
        userDAO.insert(user);
        if (!CollectionUtils.isEmpty(param.getRoleIds())) {
            userRoleService.addUserRole(user.getId(), param.getRoleIds());
        }
    }

新增用戶的同時還添加了用戶角色,這里就是使用@Transactional來控制事務保證一致性的。但大多數開發僅限于為方法標記 @Transactional來開啟聲明式事務,認為就可以高枕無憂了,不會去關注事務是否有效、出錯后事務是否正確回滾,也不會考慮復雜的業務代碼中涉及多個子業務邏輯時,怎么正確處理事務。事務沒有被正確處理,一般來說不會過于影響正常流程,也不容易在測試階段被發現。但當系統越來越復雜、壓力越來越大之后,就會帶來大量的數據不一致問題,隨后就是大量的人工介入查看和修復數據。

正是因為聲明式事務@Transactional使用簡單,所以很多開發人員不注重細節點,但是@Transactional條條框框還蠻多的,可謂是細節點拉滿,如果不注意也不小心就會掉進坑里,今天就讓我們一起來了解使用細節,把坑填平咯。

2.@Transactional

話不多說,先看看該注解定義

@Target({ElementType.TYPE, ElementType.METHOD})
@Retention(RetentionPolicy.RUNTIME)
@Inherited
@Documented
public @interface Transactional {

 @AliasFor("transactionManager")
 String value() default "";

 @AliasFor("value")
 String transactionManager() default "";

 Propagation propagation() default Propagation.REQUIRED;

 Isolation isolation() default Isolation.DEFAULT;

 int timeout() default TransactionDefinition.TIMEOUT_DEFAULT;

 boolean readOnly() default false;

 Class<? extends Throwable>[] rollbackFor() default {};

 String[] rollbackForClassName() default {};

 Class<? extends Throwable>[] noRollbackFor() default {};

 String[] noRollbackForClassName() default {};

}

從上面看出@Transactional既可以作用于類上,也可以作用于方法上,作用于類:表示所有該類的**public**方法都配置相同的事務屬性信息。接下來再看看其屬性:

propagation: 設置事務的傳播行為,主要解決是A方法調用B方法時,事務的傳播方式問題的,默認值為 **Propagation.REQUIRED**,其他屬性值信息如下:

事務傳播行為

解釋

REQUIRED(默認值)

A調用B,B需要事務,如果A有事務B就加入A的事務中,如果A沒有事務,B就自己創建一個事務

REQUIRED_NEW

A調用B,B需要新事務,如果A有事務就掛起,B自己創建一個新的事務

SUPPORTS

A調用B,B有無事務無所謂,A有事務就加入到A事務中,A無事務B就以非事務方式執行

NOT_SUPPORTS

A調用B,B以無事務方式執行,A如有事務則掛起

NEVER

A調用B,B以無事務方式執行,A如有事務則拋出異常

MANDATORY

A調用B,B要加入A的事務中,如果A無事務就拋出異常

NESTED

A調用B,B創建一個新事務,A有事務就作為嵌套事務存在,A沒事務就以創建的新事務執行

isolation :事務的隔離級別,默認值為 Isolation.DEFAULT。指定事務的隔離級別,事務并發存在三大問題:臟讀、不可重復讀、幻讀/虛讀??梢酝ㄟ^設置事務的隔離級別來保證并發問題的出現,常用的是READ_COMMITTED 和REPEATABLE_READ

isolation屬性

解釋

DEFAULT

默認隔離級別,取決于當前數據庫隔離級別,例如MySQL默認隔離級別是REPEATABLE_READ

READ_UNCOMMITTED

A事務可以讀取到B事務尚未提交的事務記錄,不能解決任何并發問題,安全性最低,性能最高

READ_COMMITTED

A事務只能讀取到其他事務已經提交的記錄,不能讀取到未提交的記錄??梢越鉀Q臟讀問題,但是不能解決不可重復讀和幻讀

REPEATABLE_READ

A事務多次從數據庫讀取某條記錄結果一致,可以解決不可重復讀,不可以解決幻讀

SERIALIZABLE

串行化,可以解決任何并發問題,安全性最高,但是性能最低

timeout :事務的超時時間,默認值為 -1。如果超過該時間限制但事務還沒有完成,則自動回滾事務。

readOnly:指定事務是否為只讀事務,默認值為 false;為了忽略那些不需要事務的方法,比如讀取數據,可以設置 read-only 為 true。

rollbackFor:用于指定能夠觸發事務回滾的異常類型,可以指定多個異常類型。

noRollbackFor:拋出指定的異常類型,不回滾事務,也可以指定多個異常類型。

項目推薦:基于SpringBoot2.x、SpringCloud和SpringCloudAlibaba企業級系統架構底層框架封裝,解決業務開發時常見的非功能性需求,防止重復造輪子,方便業務快速開發和企業技術??蚣芙y一管理。引入組件化的思想實現高內聚低耦合并且高度可配置化,做到可插拔。嚴格控制包依賴和統一版本管理,做到最少化依賴。注重代碼規范和注釋,非常適合個人學習和企業使用

Github地址:https://github.com/plasticene/plasticene-boot-starter-parent

Gitee地址:https://gitee.com/plasticene3/plasticene-boot-starter-parent

微信公眾號:Shepherd進階筆記

交流探討qun:Shepherd_126

3.@Transactional失效場景、原因及修正方式

3.1 同一個類中的方法通過this調用導致失效

public void addUser(UserParam param) {
        User user = PtcBeanUtils.copy(param, User.class);
        // 新增用戶
        userDAO.insert(user);
        // 添加用戶角色
        this.addUserRole(user.getId(), param.getRoleIds());
        log.info("執行結束了");
    }

    @Transactional(rollbackFor = Exception.class)
    public void addUserRole(Long userId, List<Long> roleIds) {
        if (CollectionUtils.isEmpty(roleIds)) {
            return;
        }
        List<UserRole> userRoles = new ArrayList<>();
        roleIds.forEach(roleId -> {
            UserRole userRole = new UserRole();
            userRole.setUserId(userId);
            userRole.setRoleId(roleId);
            userRoles.add(userRole);
        });
        userRoleDAO.insertBatch(userRoles);
        throw new RuntimeException("發生異???);
    }

執行#addUser()會發現事務控制失效,發生異常事務并沒有回滾,用戶和角色綁定都插入成功了。

這里,我給出@Transactional 生效原則 1,必須通過代理過的類從外部調用目標方法才能生效.

圖片圖片

Spring 是通過 AOP 技術對方法進行增強實現事務控制的,要調用增強過的方法必然是調用代理后的對象,而這里this是原生對象,并不是代理,自然就沒有事務控制了。

修正方式:①:將this換成代理的userService, 可以自己注入自己@Resource private UserService userService,當然也可以不用注入,直接在Spring容器中獲取userService這個bean     ②將#addUser()方法開啟事務即加上@Transactional(rollbackFor = Exception.class),這里本就該開啟,只是為了演示失效情況沒加上,因為在#addUser()里面有插入用戶的操作涉及到事務的所以本要開啟。當然如果#addUser()只是做一些判斷、邏輯處理不涉及到數據庫事務操作,那么這樣解決就顯得有點不太合適,而且容易導致另一種事務失效的情況,即因為沒有正確處理異常,導致事務即便生效也不一定能回滾。

3.2 異常被catch“吃掉了”導致@Transactional失效

如下所示:

@Transactional(rollbackFor = Exception.class)
    public void addUser(UserParam param) {
        try {
            User user = PtcBeanUtils.copy(param, User.class);
            // 完成一些邏輯處理
          
            .......
              
            // 添加用戶角色
            this.addUserRole(user.getId(), param.getRoleIds());
            log.info("執行結束了");
        } catch (Exception e) {
            log.error(e.getMessage());
        }
    }

    @Transactional(rollbackFor = Exception.class)
    public void addUserRole(Long userId, List<Long> roleIds) {
        if (CollectionUtils.isEmpty(roleIds)) {
            return;
        }
        List<UserRole> userRoles = new ArrayList<>();
        roleIds.forEach(roleId -> {
            UserRole userRole = new UserRole();
            userRole.setUserId(userId);
            userRole.setRoleId(roleId);
            userRoles.add(userRole);
        });
        userRoleDAO.insertBatch(userRoles);
        throw new RuntimeException("發生異???);
    }

@Transactional生效原則2:只有異常傳播出了標記了 @Transactional 注解的方法,事務才能回滾。之前我們總結過 基于AOP事務控制實現原理說過在 Spring的 TransactionAspectSupport 里有個 invokeWithinTransaction 方法,里面就是處理事務的邏輯??梢钥吹?,只有捕獲到異常才能進行后續事務處理:

protected Object invokeWithinTransaction(Method method, @Nullable Class<?> targetClass,
   final InvocationCallback invocation) throws Throwable {
      
      ......
        
      try {
    // This is an around advice: Invoke the next interceptor in the chain.
    // This will normally result in a target object being invoked.
    retVal = invocation.proceedWithInvocation();
   }
   catch (Throwable ex) {
    // target invocation exception
        // 捕獲到異常,進行回滾操作,如果我們在業務方法已經捕獲掉異常,這里就捕獲不到了,自然就不會回滾了
    completeTransactionAfterThrowing(txInfo, ex);
    throw ex;
   }
   finally {
    cleanupTransactionInfo(txInfo);
   }
    
    ......
        
   return result;
  }
 }

可以看到,只有捕獲到異常時才進行回滾操作,如果我們在業務方法已經捕獲掉異常,這里就捕獲不到了,自然就不會回滾了。

修正方式:就是對異常捕獲盡量做到局部針對操作,不要籠統把整個方法的代碼邏輯都包括進行,這樣異常就拋出去了。

3.3 @Transactional 屬性 rollbackFor 設置錯誤,導致異常不滿足回滾條件

直接看代碼:

@Transactional
  public void addUser(UserParam param) {
      User user = PtcBeanUtils.copy(param, User.class);
       
      .......
        
      // 添加用戶角色
      this.addUserRole(user.getId(), param.getRoleIds());
      log.info("執行結束了");
    }

    public void addUserRole(Long userId, List<Long> roleIds) throws Exception {
        if (CollectionUtils.isEmpty(roleIds)) {
            return;
        }
        List<UserRole> userRoles = new ArrayList<>();
        roleIds.forEach(roleId -> {
            UserRole userRole = new UserRole();
            userRole.setUserId(userId);
            userRole.setRoleId(roleId);
            userRoles.add(userRole);
        });
        userRoleDAO.insertBatch(userRoles);
        throw new Exception("發生異???);
    }

這里#addUser()使用@transactional,但沒有設置rollbackFor屬性,且#addUserRole()拋出的異常是exception,不是RuntimeException,這樣事務也失效了,因為默認情況下,出現 RuntimeException(非受檢異常)或 Error 的時候,Spring才會回滾事務

從上面3.2小節的completeTransactionAfterThrowing(txInfo, ex);進去完成回滾操作會判斷異常類型是否滿足規定,DefaultTransactionAttribute 類能看到如下代碼塊,可以發現相關證據,通過注釋也能看到 Spring 這么做的原因,大概的意思是受檢異常一般是業務異常,或者說是類似另一種方法的返回值,出現這樣的異??赡軜I務還能完成,所以不會主動回滾;而Error 或 RuntimeException 代表了非預期的結果,應該回滾:

public boolean rollbackOn(Throwable ex) {
  return (ex instanceof RuntimeException || ex instanceof Error);
 }

修正方法:設置rollbackFor:@Transactional(rollbackFor = Exception.class)

3.4 @Transactional 應用在非 public 修飾的方法上

@Transactional(rollbackFor = Exception.class)
    private void addUserRole(Long userId, List<Long> roleIds) {
        if (CollectionUtils.isEmpty(roleIds)) {
            return;
        }
        List<UserRole> userRoles = new ArrayList<>();
        roleIds.forEach(roleId -> {
            UserRole userRole = new UserRole();
            userRole.setUserId(userId);
            userRole.setRoleId(roleId);
            userRoles.add(userRole);
        });
        userRoleDAO.insertBatch(userRoles);
        throw new RuntimeException("發生異???);
    }

idea也會提示爆紅:

圖片圖片

Spring通過CGLIB動態代理來增強生產代理對象,CGLIB 通過繼承方式實現代理類,private 方法在子類不可見,自然也就無法進行事務增強。s在基于AOP事務控制實現原理一文中也分析過,會調用到AbstractFallbackTransactionAttributeSource的computeTransactionAttribute()方法

@Nullable
 protected TransactionAttribute computeTransactionAttribute(Method method, @Nullable Class<?> targetClass) {
    // Don't allow no-public methods as required.
    if (allowPublicMethodsOnly() && !Modifier.isPublic(method.getModifiers())) {
     return null;
    }
    
    ......
 }

修正方式:自然是改成public

3.5 @Transactional 注解傳播屬性 propagation 設置錯誤

如上面我們新增的用戶的同時要添加用戶角色,但是假如我們希望即使添加角色錯誤了,還可以正常新增用戶。

public void addUser(UserParam param) {
      String username = param.getUsername();
      checkUsernameUnique(username);
      User user = PtcBeanUtils.copy(param, User.class);
      // 添加用戶
      userDAO.insert(user);

      // 添加用戶角色
      userRoleService.addUserRole(user.getId(), param.getRoleIds());
    
 }

#userRoleService.addUserRole()

@Transactional(rollbackFor = Exception.class)
  private void addUserRole(Long userId, List<Long> roleIds) {
      if (CollectionUtils.isEmpty(roleIds)) {
          return;
      }
      List<UserRole> userRoles = new ArrayList<>();
      roleIds.forEach(roleId -> {
          UserRole userRole = new UserRole();
          userRole.setUserId(userId);
          userRole.setRoleId(roleId);
          userRoles.add(userRole);
      });
      userRoleDAO.insertBatch(userRoles);
      throw new RuntimeException("發生異???);
  }

你會發現只會同時插入失敗,無法實現上面所說的。這時候你可能會想到,既然addUserRole()拋出了異常不能插入用戶角色,但是addUser()不想受影響,正常添加用戶,那么何不在addUser()里面對userRoleService.addUserRole()進行異常捕獲,不就可以解決問題了嗎?真是如此嗎,就讓我們來驗證一下:

@Transactional(rollbackFor = Exception.class)
    public void addUser(UserParam param) {
        User user = PtcBeanUtils.copy(param, User.class);
        // 添加用戶
        userDAO.insert(user);
        // 添加用戶角色
        try {
            userRoleService.addUserRole(user.getId(), param.getRoleIds());
        } catch (Exception e) {
            log.error(e.getMessage());
        }
    }

執行會發現,用戶同樣沒有添加成功,看日志報錯:

[1689568520410750976] [ERROR] [2023-08-10 17:25:02.023] [http-nio-18888-exec-1@56682]  com.plasticene.fast.service.impl.UserServiceImpl addUser : 發生異常咯
[1689568520410750976] [ERROR] [2023-08-10 17:25:02.097] [http-nio-18888-exec-1@56682]  com.plasticene.boot.web.core.global.GlobalExceptionHandler exceptionHandler : 【系統異?!?org.springframework.transaction.UnexpectedRollbackException: Transaction rolled back because it has been marked as rollback-only
 at org.springframework.transaction.support.AbstractPlatformTransactionManager.processRollback(AbstractPlatformTransactionManager.java:870)

可以看到發生異常咯是我們在addUser()中捕獲到輸出的,但是緊接著下一行發現有報出一個異常UnexpectedRollbackException。

原因是,主方法添加用戶的邏輯和子方法添加用戶角色的邏輯是同一個事務,子邏輯標記了事務需要回滾,主邏輯自然也不能提交了。

修正方式:其實要想新增用戶角色失敗不影響添加用戶,只需要讓新增用戶角色單獨開啟一個新事務即可。

@Transactional(rollbackFor = Exception.class, propagation = Propagation.REQUIRES_NEW)
    public void addUserRole(Long userId, List<Long> roleIds) {
        List<UserRole> userRoles = new ArrayList<>();
        roleIds.forEach(roleId -> {
            UserRole userRole = new UserRole();
            userRole.setUserId(userId);
            userRole.setRoleId(roleId);
            userRoles.add(userRole);
        });
        userRoleDAO.insertBatch(userRoles);
        throw new RuntimeException("發生異常啦!");
    }

3.6 @Transactional長事務導致生產事故

很多開發都覺得Spring的聲明式事務使用非常簡單,即@Transactional,所以從來不注重細節。當 Spring 遇到該注解時,會自動從數據庫連接池中獲取 connection,并開啟事務然后綁定到 ThreadLocal 上,對于@Transactional注解包裹的整個方法都是使用同一個connection連接。如果我們出現了耗時的操作,比如第三方接口調用、業務邏輯復雜、大批量數據處理等就會導致我們我們占用這個connection的時間會很長,數據庫連接一直被占用不釋放。一旦類似操作過多,就會導致數據庫連接池耗盡。這就是典型的長事務問題

長事務引發的常見危害有:

  1. 數據庫連接池被占滿,應用無法獲取連接資源;
  2. 容易引發數據庫死鎖;
  3. 數據庫回滾時間長;
  4. 在主從架構中會導致主從延時變大。

服務系統開始出現故障:數據庫監控平臺一直收到告警短信,數據庫連接不足,出現大量死鎖;日志顯示調用流程引擎接口出現大量超時;同時一直提示CannotGetJdbcConnectionException,數據庫連接池連接占滿。

要想解決這個問題其實也不難,只需要對方法進行拆分,將不需要事務管理的邏輯與事務操作分開,這樣就可以有效控制事務的時長從而避免長事務。當然對一個方法邏輯拆分成多個子方法很有可能造成上面敘述的事務不生效的情況,不過我相信你看到上面的總結肯定沒問題啦。

4.總結

Spring的聲明式事務使用@Transactional注解在開發時確實很方便,但是稍有不慎使用不當就會導致事務失效數據不一致、甚至是系統數據庫性能問題。所以上面滿滿的干貨總結都是出自日常工作中碰到的,有效幫你避坑。

本文轉載自微信公眾號「Shepherd進階筆記」,可以通過以下二維碼關注。轉載本文請聯系公眾號。


責任編輯:武曉燕 來源: Shepherd進階筆記
相關推薦

2023-09-08 08:52:12

Spring注解事務

2022-08-08 17:38:45

Spring策略事務

2022-08-09 09:34:32

Spring開發

2023-09-28 09:07:54

注解失效場景

2022-07-27 10:39:14

Spring代碼IDEA

2022-04-26 21:49:55

Spring事務數據庫

2023-09-27 16:22:51

SpringMySQL原子性

2024-01-30 08:01:15

RabbitMQ業務邏輯應用場景

2025-06-09 07:38:23

2024-10-06 08:23:28

2025-08-26 01:20:00

2022-09-20 22:27:08

事務失效public 修飾

2025-02-10 00:27:54

2021-09-04 07:56:44

Spring事務失效

2023-05-05 07:39:04

Spring事務面試

2021-08-04 11:05:19

B端C端設計

2024-01-29 08:28:01

Spring事務失效

2022-09-14 19:50:22

事務場景流程

2022-02-14 16:53:57

Spring項目數據庫

2020-11-24 11:30:51

SpringJava代碼
點贊
收藏

51CTO技術棧公眾號

日本免费新一区视频| 亚洲一区二区三区日本久久九| 久久精品一区二区三区不卡| 日韩av黄色在线观看| 久久久久久久久久久久久久久| 黑人巨大亚洲一区二区久| 久久久久久电影| 国产精品91一区| 国产午夜精品理论片在线| 日本精品国产| 欧美午夜女人视频在线| 污视频在线免费观看一区二区三区| 影音先锋国产资源| 一区二区三区午夜探花| 精品国产一区二区精华| 日日橹狠狠爱欧美超碰| 99免在线观看免费视频高清| 狠狠色丁香久久婷婷综| 亚洲91精品在线| av男人的天堂av| 国产高清亚洲| 日韩欧美国产成人| 一区二区三区国| 粉嫩av一区二区夜夜嗨| 日本午夜精品一区二区三区电影| 欧美成人精品在线观看| 右手影院亚洲欧美| 精品国模一区二区三区欧美 | 91精品人妻一区二区三区| 国产精品诱惑| 天天综合网天天综合色| 一区二区三区我不卡| 熟妇高潮一区二区高潮| 精品在线观看视频| 欧美资源在线观看| 国产一区二区播放| 教室别恋欧美无删减版| 欧美精品一区二区三区蜜臀| 黄色片视频在线| 国产传媒在线观看| 亚洲欧美日韩中文播放| 日本成人看片网址| 人妻精品无码一区二区| 国内精品伊人久久久久av影院 | 国产精品视频你懂的| 国产一区免费观看| 国产麻豆免费观看| 日韩成人一区二区| 91极品女神在线| 印度午夜性春猛xxx交| 国产va免费精品观看精品视频| 欧美大片一区二区| 911av视频| 国产亚洲精彩久久| 欧美中文字幕一区二区三区亚洲| 黄色免费福利视频| 高清精品在线| 亚洲国产成人精品视频| 国产a级黄色大片| 九色porny丨首页在线| 欧美极品aⅴ影院| 欧美精品七区| 黑人精品一区二区三区| 国产一区二区三区日韩| 91精品中国老女人| 中文字幕一区二区人妻视频| 亚洲在线视频| 欧美高清在线播放| 紧身裙女教师波多野结衣| 久久亚洲国产| 亚洲精品一区二区三区婷婷月| 噜噜噜在线视频| 神马香蕉久久| 亚洲香蕉av在线一区二区三区| www.色天使| 国产欧美日韩在线一区二区| 伊人男人综合视频网| 熟女少妇内射日韩亚洲| 成人影院在线| 日韩在线播放一区| 久久久久久视频| 婷婷久久一区| 欧美黑人xxxx| 日本亚洲欧美在线| 国产亚洲综合精品| 国产精品福利在线观看网址| 少妇一级淫片日本| 国产一区中文字幕| 国产精品乱子乱xxxx| 五月婷婷伊人网| 国产日韩欧美a| 一区二区三视频| av超碰免费在线| 精品国产乱码久久久久久虫虫漫画| 一区二区传媒有限公司| 人人妻人人爽人人澡人人精品| 亚洲欧洲一区| 日韩av片免费在线观看| 这里只有精品9| 国产精品自拍一区| 国产区日韩欧美| 极品白浆推特女神在线观看| 中文字幕在线观看不卡视频| 青青视频免费在线| 天天综合av| 欧美日韩国产高清一区| 精品无码av一区二区三区不卡| 看全色黄大色大片免费久久久| 亚洲欧美国产精品久久久久久久| 爱爱免费小视频| 99视频精品全部免费在线视频| 欧美理论电影在线观看| 国产成人精品亚洲男人的天堂| 首页国产欧美日韩丝袜| 亚洲一区二区免费在线| 天堂在线一二区| 1000部国产精品成人观看| 久久久久久人妻一区二区三区| 高清电影一区| 日韩女同互慰一区二区| 亚洲色成人网站www永久四虎 | 欧美国产日韩在线观看成人| 亚洲一区区二区| 亚洲综合精品一区二区| 欧美少妇另类| 亚洲精品日韩专区silk| av动漫免费看| jazzjazz国产精品久久| 日日噜噜噜夜夜爽亚洲精品| 精品美女久久久久| 国产精品一区在线| 日韩欧美国产二区| 国产蜜臀av在线播放| 欧美日韩一区成人| 免费在线观看你懂的| 国产精品分类| 成人在线播放av| 高清日韩av电影| 精品国产老师黑色丝袜高跟鞋| 中文 日韩 欧美| 欧洲杯半决赛直播| 日韩av手机在线| 天天干天天摸天天操| 亚洲综合无码一区二区| 亚洲精品免费一区亚洲精品免费精品一区 | 欧美成人午夜免费视在线看片| 啪啪小视频网站| 91麻豆国产在线观看| 日本福利视频一区| 中文一区二区三区四区| 久久综合88中文色鬼| 中文字幕永久在线观看| 久久精品一级爱片| 欧美日韩精品一区| 后进极品白嫩翘臀在线播放| 欧美夫妻性生活| 国产喷水在线观看| 美女视频网站久久| 亚洲 国产 日韩 综合一区| 香蕉伊大人中文在线观看| 欧美不卡一区二区三区四区| 欧美国产日韩在线观看成人| 激情亚洲综合在线| 超碰10000| 久久久久久亚洲精品美女| www.久久久久久.com| 亚洲网站在线免费观看| 国产精品女同一区二区三区| 欧美日韩在线成人| 欧美中文一区二区| 国产精品一区二区久久久| 国产日本在线视频| 欧美午夜免费电影| 黄色激情小视频| 美女视频一区二区三区| 亚洲欧美一二三| 亚洲啊v在线免费视频| 欧美极品欧美精品欧美视频 | 黄色精品一区| 国产一区二区三区无遮挡| 蜜桃av在线播放| 亚洲人成在线播放| 一区二区自拍偷拍| 亚洲少妇中出一区| 无码国产精品一区二区高潮| 伊人久久亚洲美女图片| 久久本道综合色狠狠五月| 粉嫩一区二区| 色哟哟入口国产精品| 一区二区三区免费在线| 一区二区三区精品| 蜜桃精品成人影片| 免费观看30秒视频久久| 国产精品亚洲天堂| 另类在线视频| 国产精品视频久久久| gogo在线高清视频| 亚洲激情在线观看| 销魂美女一区二区| 亚洲美女在线国产| 在线观看日韩精品视频| 男男成人高潮片免费网站| 看全色黄大色大片| 亚洲成aⅴ人片久久青草影院| 国产精品视频午夜| 精精国产xxxx视频在线中文版 | 欧美一级xxxx| 亚洲美女黄色| 一本色道久久综合亚洲精品婷婷| 日韩一区免费| 国产精品com| 三级福利片在线观看| 亚洲午夜小视频| 亚洲AV无码精品自拍| 色综合久久综合中文综合网| 国产探花视频在线播放| 国产福利一区二区| 少妇一级淫免费放| 影音先锋久久资源网| 在线免费一区| 亚洲天堂日韩在线| 91九色偷拍| 成人在线高清| 欧美重口另类videos人妖| 91精品久久久久久粉嫩| 亚洲色图日韩av| 天天av天天翘| 日韩免费性生活视频播放| 中文亚洲av片在线观看| 午夜国产精品一区| 欧美日韩在线观看成人| 国产日韩精品一区| 国产又黄又粗又猛又爽的视频| 精油按摩中文字幕久久| 国产亚洲精品网站| 国内精品福利| 日韩精品久久久毛片一区二区| 国产精品115| 成人av番号网| 少妇视频在线观看| 欧美激情一级欧美精品| 国产黄色在线观看| 中文字幕精品在线视频| 男女av在线| 精品调教chinesegay| 亚洲AV无码精品自拍| 欧美色爱综合网| 无码人妻精品一区二区三区9厂| 欧美日韩美女在线观看| 日本少妇性生活| 亚洲成人动漫精品| 国产这里有精品| 亚洲另类一区二区| www欧美com| 亚洲男人的天堂网| 在线免费看av网站| 亚洲图片欧美激情| 国产97免费视频| 亚洲黄色免费网站| 精品国产一区a| 精品国产一区二区三区四| 亚洲蜜臀av乱码久久精品| 日韩欧美综合视频| 亚洲黄色免费网站| 久久影院一区二区| 亚洲国产一区二区三区青草影视| 欧美色图一区二区| 亚洲一区二区三区视频在线播放| 青娱乐国产在线| 亚洲成av人在线观看| 日韩欧美三级在线观看| 精品久久久免费| 久久亚洲av午夜福利精品一区| 一区二区三区在线播| 丰满少妇高潮久久三区| 亚洲电影在线播放| 日韩精品手机在线| 欧洲生活片亚洲生活在线观看| 国产在线一级片| 欧美久久久久久久久久| 国内精品偷拍视频| 亚洲激情自拍图| 国产高清免费av在线| www.日韩av.com| 欧美色图天堂| 青草热久免费精品视频| 91在线亚洲| 亚洲在线观看视频| 国产伦精品一区二区三区在线播放 | 色呦呦网站在线观看| 97热精品视频官网| 91精品影视| 91夜夜揉人人捏人人添红杏| 伊人精品久久| 日本中文不卡| 午夜精品av| 欧美一区二区三区爽大粗免费| 日韩中文字幕亚洲一区二区va在线 | 少妇搡bbbb搡bbb搡打电话| 91亚洲永久精品| 亚洲精品自拍视频在线观看| 亚洲一区二区三区四区的| 国产一级免费视频| 欧美一区二区久久| 欧美女v视频| 欧美超级乱淫片喷水| 周于希免费高清在线观看| 国产在线视频欧美| 亚州综合一区| 婷婷精品国产一区二区三区日韩| 欧美三区不卡| 五月婷婷六月合| 国产成人av影院| 伊人网伊人影院| 亚洲精品伦理在线| 亚洲精品无码久久久久| 亚洲黄页网在线观看| 岛国成人毛片| 国产成人鲁鲁免费视频a| 99久热这里只有精品视频免费观看| 国产欧美日韩一区二区三区| 欧美aaaaaaaaaaaa| 91传媒久久久| 国产精品中文字幕日韩精品| 受虐m奴xxx在线观看| 亚洲欧美电影一区二区| 精品久久久久久久久久久久久久久久| 日韩欧美国产一区二区在线播放| 成人免费一区二区三区视频网站| 欧美理论片在线观看| 日韩国产大片| 日韩国产精品一区二区三区| 日韩视频中文| 波多野结衣三级视频| 国产精品国产a| 成人毛片一区二区三区| 亚洲精品成人久久电影| 色www永久免费视频首页在线 | 亚洲第一福利在线观看| www.久久ai| 成人黄色短视频在线观看| 国产精品密蕾丝视频下载| 成年人午夜免费视频| 国产**成人网毛片九色| 91九色丨porny丨极品女神| 欧美体内she精视频| 久久精品蜜桃| 国产v综合v亚洲欧美久久| 久久a爱视频| 欧美中日韩在线| 国产成人免费网站| 我家有个日本女人| 日韩欧美中文字幕公布| 97超碰资源站在线观看| 91香蕉亚洲精品| 自产国语精品视频| 先锋资源在线视频| 一区二区三区四区在线免费观看| 国产麻豆免费观看| 九九九久久国产免费| 欧美xxxx网站| 一区二区三区我不卡| 精品一区二区三区免费毛片爱| 妺妺窝人体色WWW精品| 婷婷久久综合九色综合伊人色| 欧美自拍偷拍第一页| 97碰在线观看| 嫩草影视亚洲| av污在线观看| 自拍av一区二区三区| www.久久色| 久久免费福利视频| 红杏一区二区三区| 少妇人妻大乳在线视频| 成人午夜免费av| 久久久久久久久久久97| 日韩一级片在线播放| 日本三级在线观看网站| 精品国产一区二区三区麻豆小说 | 你懂的亚洲视频| 国产不卡一二三| 在线看国产日韩| 欧美videossex| 热re99久久精品国产99热| 精品一区二区三区日韩| 亚洲欧美在线视频免费| 自拍偷拍亚洲在线| 黄色成人美女网站| 91国内在线播放| 偷窥少妇高潮呻吟av久久免费| 9191在线| 好吊色欧美一区二区三区四区| 免费精品99久久国产综合精品| 精品少妇theporn| 深夜福利一区二区| 日本午夜精品| 伊人av在线播放| 欧美日韩精品三区| 中文字幕成在线观看| 成人国产一区二区三区|