成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

工作幾年了,原來我只用了數(shù)據(jù)校驗(yàn)的皮毛

開發(fā) 前端
不知不覺Spring Boot專欄文章已經(jīng)寫到第十四章了,無論寫的好與不好,作者都在盡力寫的詳細(xì),寫的與其它的文章不同,每一章都不是淺嘗輒止。

[[433466]]

前言

不知不覺Spring Boot專欄文章已經(jīng)寫到第十四章了,無論寫的好與不好,作者都在盡力寫的詳細(xì),寫的與其它的文章不同,每一章都不是淺嘗輒止。如果前面的文章沒有看過的朋友,點(diǎn)擊這里前往。

今天介紹一下 Spring Boot 如何優(yōu)雅的整合JSR-303進(jìn)行參數(shù)校驗(yàn),說到參數(shù)校驗(yàn)可能都用過,但是你真的會(huì)用嗎?網(wǎng)上的教程很多,大多是簡(jiǎn)單的介紹。

什么是 JSR-303?

JSR-303 是 JAVA EE 6 中的一項(xiàng)子規(guī)范,叫做 Bean Validation。

Bean Validation 為 JavaBean 驗(yàn)證定義了相應(yīng)的元數(shù)據(jù)模型和API。缺省的元數(shù)據(jù)是Java Annotations,通過使用 XML 可以對(duì)原有的元數(shù)據(jù)信息進(jìn)行覆蓋和擴(kuò)展。在應(yīng)用程序中,通過使用Bean Validation 或是你自己定義的 constraint,例如 @NotNull, @Max, @ZipCode , 就可以確保數(shù)據(jù)模型(JavaBean)的正確性。constraint 可以附加到字段,getter 方法,類或者接口上面。對(duì)于一些特定的需求,用戶可以很容易的開發(fā)定制化的 constraint。Bean Validation 是一個(gè)運(yùn)行時(shí)的數(shù)據(jù)驗(yàn)證框架,在驗(yàn)證之后驗(yàn)證的錯(cuò)誤信息會(huì)被馬上返回。

添加依賴

Spring Boot整合JSR-303只需要添加一個(gè)starter即可,如下:

  1. <dependency> 
  2.     <groupId>org.springframework.boot</groupId> 
  3.    <artifactId>spring-boot-starter-validation</artifactId> 
  4. </dependency> 

內(nèi)嵌的注解有哪些?

Bean Validation 內(nèi)嵌的注解很多,基本實(shí)際開發(fā)中已經(jīng)夠用了,注解如下:

注解 詳細(xì)信息
@Null 被注釋的元素必須為 null
@NotNull 被注釋的元素必須不為 null
@AssertTrue 被注釋的元素必須為 true
@AssertFalse 被注釋的元素必須為 false
@Min(value) 被注釋的元素必須是一個(gè)數(shù)字,其值必須大于等于指定的最小值
@Max(value) 被注釋的元素必須是一個(gè)數(shù)字,其值必須小于等于指定的最大值
@DecimalMin(value) 被注釋的元素必須是一個(gè)數(shù)字,其值必須大于等于指定的最小值
@DecimalMax(value) 被注釋的元素必須是一個(gè)數(shù)字,其值必須小于等于指定的最大值
@Size(max, min) 被注釋的元素的大小必須在指定的范圍內(nèi)
@Digits (integer, fraction) 被注釋的元素必須是一個(gè)數(shù)字,其值必須在可接受的范圍內(nèi)
@Past 被注釋的元素必須是一個(gè)過去的日期
@Future 被注釋的元素必須是一個(gè)將來的日期
@Pattern(value) 被注釋的元素必須符合指定的正則表達(dá)式

以上是Bean Validation的內(nèi)嵌的注解,但是Hibernate Validator在原有的基礎(chǔ)上也內(nèi)嵌了幾個(gè)注解,如下。

注解 詳細(xì)信息
@Email 被注釋的元素必須是電子郵箱地址
@Length 被注釋的字符串的大小必須在指定的范圍內(nèi)
@NotEmpty 被注釋的字符串的必須非空
@Range 被注釋的元素必須在合適的范圍內(nèi)

如何使用?

參數(shù)校驗(yàn)分為簡(jiǎn)單校驗(yàn)、嵌套校驗(yàn)、分組校驗(yàn)。

簡(jiǎn)單校驗(yàn)

簡(jiǎn)單的校驗(yàn)即是沒有嵌套屬性,直接在需要的元素上標(biāo)注約束注解即可。如下:

  1. @Data 
  2. public class ArticleDTO { 
  3.  
  4.     @NotNull(message = "文章id不能為空"
  5.     @Min(value = 1,message = "文章ID不能為負(fù)數(shù)"
  6.     private Integer id; 
  7.  
  8.     @NotBlank(message = "文章內(nèi)容不能為空"
  9.     private String content; 
  10.  
  11.     @NotBlank(message = "作者Id不能為空"
  12.     private String authorId; 
  13.  
  14.     @Future(message = "提交時(shí)間不能為過去時(shí)間"
  15.     private Date submitTime; 

同一個(gè)屬性可以指定多個(gè)約束,比如@NotNull和@MAX,其中的message屬性指定了約束條件不滿足時(shí)的提示信息。

以上約束標(biāo)記完成之后,要想完成校驗(yàn),需要在controller層的接口標(biāo)注@Valid注解以及聲明一個(gè)BindingResult類型的參數(shù)來接收校驗(yàn)的結(jié)果。

下面簡(jiǎn)單的演示下添加文章的接口,如下:

  1. /** 
  2.      * 添加文章 
  3.      */ 
  4.     @PostMapping("/add"
  5.     public String add(@Valid @RequestBody ArticleDTO articleDTO, BindingResult bindingResult) throws JsonProcessingException { 
  6.         //如果有錯(cuò)誤提示信息 
  7.         if (bindingResult.hasErrors()) { 
  8.             Map<String , String> map = new HashMap<>(); 
  9.             bindingResult.getFieldErrors().forEach( (item) -> { 
  10.                 String message = item.getDefaultMessage(); 
  11.                 String field = item.getField(); 
  12.                 map.put( field , message ); 
  13.             } ); 
  14.             //返回提示信息 
  15.             return objectMapper.writeValueAsString(map); 
  16.         } 
  17.         return "success"
  18.     } 

僅僅在屬性上添加了約束注解還不行,還需在接口參數(shù)上標(biāo)注@Valid注解并且聲明一個(gè)BindingResult類型的參數(shù)來接收校驗(yàn)結(jié)果。

分組校驗(yàn)

舉個(gè)栗子:上傳文章不需要傳文章ID,但是修改文章需要上傳文章ID,并且用的都是同一個(gè)DTO接收參數(shù),此時(shí)的約束條件該如何寫呢?

此時(shí)就需要對(duì)這個(gè)文章ID進(jìn)行分組校驗(yàn),上傳文章接口是一個(gè)分組,不需要執(zhí)行@NotNull校驗(yàn),修改文章的接口是一個(gè)分組,需要執(zhí)行@NotNull的校驗(yàn)。

所有的校驗(yàn)注解都有一個(gè)groups屬性用來指定分組,Class[]類型,沒有實(shí)際意義,因此只需要定義一個(gè)或者多個(gè)接口用來區(qū)分即可。

  1. @Data 
  2. public class ArticleDTO { 
  3.  
  4.     /** 
  5.      * 文章ID只在修改的時(shí)候需要檢驗(yàn),因此指定groups為修改的分組 
  6.      */ 
  7.     @NotNull(message = "文章id不能為空",groups = UpdateArticleDTO.class ) 
  8.     @Min(value = 1,message = "文章ID不能為負(fù)數(shù)",groups = UpdateArticleDTO.class) 
  9.     private Integer id; 
  10.  
  11.     /** 
  12.      * 文章內(nèi)容添加和修改都是必須校驗(yàn)的,groups需要指定兩個(gè)分組 
  13.      */ 
  14.     @NotBlank(message = "文章內(nèi)容不能為空",groups = {AddArticleDTO.class,UpdateArticleDTO.class}) 
  15.     private String content; 
  16.  
  17.     @NotBlank(message = "作者Id不能為空",groups = AddArticleDTO.class) 
  18.     private String authorId; 
  19.  
  20.     /** 
  21.      * 提交時(shí)間是添加和修改都需要校驗(yàn)的,因此指定groups兩個(gè) 
  22.      */ 
  23.     @Future(message = "提交時(shí)間不能為過去時(shí)間",groups = {AddArticleDTO.class,UpdateArticleDTO.class}) 
  24.     private Date submitTime; 
  25.      
  26.     //修改文章的分組 
  27.     public interface UpdateArticleDTO{} 
  28.  
  29.     //添加文章的分組 
  30.     public interface AddArticleDTO{} 
  31.  

JSR303本身的@Valid并不支持分組校驗(yàn),但是Spring在其基礎(chǔ)提供了一個(gè)注解@Validated支持分組校驗(yàn)。@Validated這個(gè)注解value屬性指定需要校驗(yàn)的分組。

  1. /** 
  2.      * 添加文章 
  3.      * @Validated:這個(gè)注解指定校驗(yàn)的分組信息 
  4.      */ 
  5.     @PostMapping("/add"
  6.     public String add(@Validated(value = ArticleDTO.AddArticleDTO.class) @RequestBody ArticleDTO articleDTO, BindingResult bindingResult) throws JsonProcessingException { 
  7.         //如果有錯(cuò)誤提示信息 
  8.         if (bindingResult.hasErrors()) { 
  9.             Map<String , String> map = new HashMap<>(); 
  10.             bindingResult.getFieldErrors().forEach( (item) -> { 
  11.                 String message = item.getDefaultMessage(); 
  12.                 String field = item.getField(); 
  13.                 map.put( field , message ); 
  14.             } ); 
  15.             //返回提示信息 
  16.             return objectMapper.writeValueAsString(map); 
  17.         } 
  18.         return "success"
  19.     } 

嵌套校驗(yàn)

嵌套校驗(yàn)簡(jiǎn)單的解釋就是一個(gè)實(shí)體中包含另外一個(gè)實(shí)體,并且這兩個(gè)或者多個(gè)實(shí)體都需要校驗(yàn)。

舉個(gè)栗子:文章可以有一個(gè)或者多個(gè)分類,作者在提交文章的時(shí)候必須指定文章分類,而分類是單獨(dú)一個(gè)實(shí)體,有分類ID、名稱等等。大致的結(jié)構(gòu)如下:

  1. public class ArticleDTO{ 
  2.   ...文章的一些屬性..... 
  3.    
  4.   //分類的信息 
  5.   private CategoryDTO categoryDTO; 

此時(shí)文章和分類的屬性都需要校驗(yàn),這種就叫做嵌套校驗(yàn)。

嵌套校驗(yàn)很簡(jiǎn)單,只需要在嵌套的實(shí)體屬性標(biāo)注@Valid注解,則其中的屬性也將會(huì)得到校驗(yàn),否則不會(huì)校驗(yàn)。

如下文章分類實(shí)體類校驗(yàn):

  1. /** 
  2.  * 文章分類 
  3.  */ 
  4. @Data 
  5. public class CategoryDTO { 
  6.     @NotNull(message = "分類ID不能為空"
  7.     @Min(value = 1,message = "分類ID不能為負(fù)數(shù)"
  8.     private Integer id; 
  9.  
  10.     @NotBlank(message = "分類名稱不能為空"
  11.     private String name

文章的實(shí)體類中有個(gè)嵌套的文章分類CategoryDTO屬性,需要使用@Valid標(biāo)注才能嵌套校驗(yàn),如下:

  1. @Data 
  2. public class ArticleDTO { 
  3.  
  4.     @NotBlank(message = "文章內(nèi)容不能為空"
  5.     private String content; 
  6.  
  7.     @NotBlank(message = "作者Id不能為空"
  8.     private String authorId; 
  9.  
  10.     @Future(message = "提交時(shí)間不能為過去時(shí)間"
  11.     private Date submitTime; 
  12.  
  13.     /** 
  14.      * @Valid這個(gè)注解指定CategoryDTO中的屬性也需要校驗(yàn) 
  15.      */ 
  16.     @Valid 
  17.     @NotNull(message = "分類不能為空"
  18.     private CategoryDTO categoryDTO; 
  19.   } 

Controller層的添加文章的接口同上,需要使用@Valid或者@Validated標(biāo)注入?yún)ⅲ瑫r(shí)需要定義一個(gè)BindingResult的參數(shù)接收校驗(yàn)結(jié)果。

嵌套校驗(yàn)針對(duì)分組查詢?nèi)匀簧В绻短椎膶?shí)體類(比如CategoryDTO)中的校驗(yàn)的屬性和接口中@Validated注解指定的分組不同,則不會(huì)校驗(yàn)。

JSR-303針對(duì)集合的嵌套校驗(yàn)也是可行的,比如List的嵌套校驗(yàn),同樣需要在屬性上標(biāo)注一個(gè)@Valid注解才會(huì)生效,如下:

  1. @Data 
  2. public class ArticleDTO { 
  3.     /** 
  4.      * @Valid這個(gè)注解標(biāo)注在集合上,將會(huì)針對(duì)集合中每個(gè)元素進(jìn)行校驗(yàn) 
  5.      */ 
  6.     @Valid 
  7.     @Size(min = 1,message = "至少一個(gè)分類"
  8.     @NotNull(message = "分類不能為空"
  9.     private List<CategoryDTO> categoryDTOS; 
  10.   } 

總結(jié):嵌套校驗(yàn)只需要在需要校驗(yàn)的元素(單個(gè)或者集合)上添加@Valid注解,接口層需要使用@Valid或者@Validated注解標(biāo)注入?yún)ⅰ?/p>

如何接收校驗(yàn)結(jié)果?

接收校驗(yàn)的結(jié)果的方式很多,不過實(shí)際開發(fā)中最好選擇一個(gè)優(yōu)雅的方式,下面介紹常見的兩種方式。

BindingResult 接收

這種方式需要在Controller層的每個(gè)接口方法參數(shù)中指定,Validator會(huì)將校驗(yàn)的信息自動(dòng)封裝到其中。這也是上面例子中一直用的方式。如下:

  1. @PostMapping("/add"
  2.     public String add(@Valid @RequestBody ArticleDTO articleDTO, BindingResult bindingResult){} 

這種方式的弊端很明顯,每個(gè)接口方法參數(shù)都要聲明,同時(shí)每個(gè)方法都要處理校驗(yàn)信息,顯然不現(xiàn)實(shí),舍棄。

此種方式還有一個(gè)優(yōu)化的方案:使用AOP,在Controller接口方法執(zhí)行之前處理BindingResult的消息提示,不過這種方案仍然不推薦使用。

全局異常捕捉

參數(shù)在校驗(yàn)失敗的時(shí)候會(huì)拋出的MethodArgumentNotValidException或者BindException兩種異常,可以在全局的異常處理器中捕捉到這兩種異常,將提示信息或者自定義信息返回給客戶端。

全局異常捕捉之前有單獨(dú)寫過一篇文章,不理解的可以看滿屏的try-catch,你不瘆得慌?。

作者這里就不再詳細(xì)的貼出其他的異常捕獲了,僅僅貼一下參數(shù)校驗(yàn)的異常捕獲(僅僅舉個(gè)例子,具體的返回信息需要自己封裝),如下:

  1. @RestControllerAdvice 
  2. public class ExceptionRsHandler { 
  3.  
  4.     @Autowired 
  5.     private ObjectMapper objectMapper; 
  6.  
  7.     /** 
  8.      * 參數(shù)校驗(yàn)異常步驟 
  9.      */ 
  10.     @ExceptionHandler(value= {MethodArgumentNotValidException.class , BindException.class}) 
  11.     public String onException(Exception e) throws JsonProcessingException { 
  12.         BindingResult bindingResult = null
  13.         if (e instanceof MethodArgumentNotValidException) { 
  14.             bindingResult = ((MethodArgumentNotValidException)e).getBindingResult(); 
  15.         } else if (e instanceof BindException) { 
  16.             bindingResult = ((BindException)e).getBindingResult(); 
  17.         } 
  18.         Map<String,String> errorMap = new HashMap<>(16); 
  19.         bindingResult.getFieldErrors().forEach((fieldError)-> 
  20.                 errorMap.put(fieldError.getField(),fieldError.getDefaultMessage()) 
  21.         ); 
  22.         return objectMapper.writeValueAsString(errorMap); 
  23.     } 
  24.  

spring-boot-starter-validation做了什么?

這個(gè)啟動(dòng)器的自動(dòng)配置類是ValidationAutoConfiguration,最重要的代碼就是注入了一個(gè)Validator(校驗(yàn)器)的實(shí)現(xiàn)類,代碼如下:

  1. @Bean 
  2.  @Role(BeanDefinition.ROLE_INFRASTRUCTURE) 
  3.  @ConditionalOnMissingBean(Validator.class) 
  4.  public static LocalValidatorFactoryBean defaultValidator() { 
  5.   LocalValidatorFactoryBean factoryBean = new LocalValidatorFactoryBean(); 
  6.   MessageInterpolatorFactory interpolatorFactory = new MessageInterpolatorFactory(); 
  7.   factoryBean.setMessageInterpolator(interpolatorFactory.getObject()); 
  8.   return factoryBean; 
  9.  } 

這個(gè)有什么用呢?Validator這個(gè)接口定義了校驗(yàn)的方法,如下:

  1. <T> Set<ConstraintViolation<T>> validate(T object, Class<?>... groups); 
  2.  
  3.  
  4. <T> Set<ConstraintViolation<T>> validateProperty(T object, 
  5.               String propertyName, 
  6.               Class<?>... groups); 
  7.                             
  8. <T> Set<ConstraintViolation<T>> validateValue(Class<T> beanType, 
  9.               String propertyName, 
  10.               Object value, 
  11.               Class<?>... groups); 
  12. ...... 

這個(gè)Validator可以用來自定義實(shí)現(xiàn)自己的校驗(yàn)邏輯,有些大公司完全不用JSR-303提供的@Valid注解,而是有一套自己的實(shí)現(xiàn),其實(shí)本質(zhì)就是利用Validator這個(gè)接口的實(shí)現(xiàn)。

如何自定義校驗(yàn)?

雖說在日常的開發(fā)中內(nèi)置的約束注解已經(jīng)夠用了,但是仍然有些時(shí)候不能滿足需求,需要自定義一些校驗(yàn)約束。

舉個(gè)栗子:有這樣一個(gè)例子,傳入的數(shù)字要在列舉的值范圍中,否則校驗(yàn)失敗。

自定義校驗(yàn)注解

首先需要自定義一個(gè)校驗(yàn)注解,如下:

  1. @Documented 
  2. @Constraint(validatedBy = { EnumValuesConstraintValidator.class}) 
  3. @Target({ METHOD, FIELD, ANNOTATION_TYPE }) 
  4. @Retention(RUNTIME) 
  5. @NotNull(message = "不能為空"
  6. public @interface EnumValues { 
  7.     /** 
  8.      * 提示消息 
  9.      */ 
  10.     String message() default "傳入的值不在范圍內(nèi)"
  11.  
  12.     /** 
  13.      * 分組 
  14.      * @return 
  15.      */ 
  16.     Class<?>[] groups() default { }; 
  17.  
  18.     Class<? extends Payload>[] payload() default { }; 
  19.  
  20.     /** 
  21.      * 可以傳入的值 
  22.      * @return 
  23.      */ 
  24.     int[] values() default { }; 

根據(jù)Bean Validation API 規(guī)范的要求有如下三個(gè)屬性是必須的:

  • message:定義消息模板,校驗(yàn)失敗時(shí)輸出
  • groups:用于校驗(yàn)分組
  • payload:Bean Validation API 的使用者可以通過此屬性來給約束條件指定嚴(yán)重級(jí)別. 這個(gè)屬性并不被API自身所使用。

除了以上三個(gè)必須要的屬性,添加了一個(gè)values屬性用來接收限制的范圍。

該校驗(yàn)注解頭上標(biāo)注的如下一行代碼:

  1. @Constraint(validatedBy = { EnumValuesConstraintValidator.class}) 

這個(gè)@Constraint注解指定了通過哪個(gè)校驗(yàn)器去校驗(yàn)。

自定義校驗(yàn)注解可以復(fù)用內(nèi)嵌的注解,比如@EnumValues注解頭上標(biāo)注了一個(gè)@NotNull注解,這樣@EnumValues就兼具了@NotNull的功能。

自定義校驗(yàn)器

@Constraint注解指定了校驗(yàn)器為EnumValuesConstraintValidator,因此需要自定義一個(gè)。

自定義校驗(yàn)器需要實(shí)現(xiàn)ConstraintValidator這個(gè)接口,第一個(gè)泛型是校驗(yàn)注解,第二個(gè)是參數(shù)類型。代碼如下:

  1. /** 
  2.  * 校驗(yàn)器 
  3.  */ 
  4. public class EnumValuesConstraintValidator implements ConstraintValidator<EnumValues,Integer> { 
  5.     /** 
  6.      * 存儲(chǔ)枚舉的值 
  7.      */ 
  8.     private  Set<Integer> ints=new HashSet<>(); 
  9.  
  10.     /** 
  11.      * 初始化方法 
  12.      * @param enumValues 校驗(yàn)的注解 
  13.      */ 
  14.     @Override 
  15.     public void initialize(EnumValues enumValues) { 
  16.         for (int value : enumValues.values()) { 
  17.             ints.add(value); 
  18.         } 
  19.     } 
  20.  
  21.     /** 
  22.      * 
  23.      * @param value  入?yún)鞯闹?nbsp;
  24.      * @param context 
  25.      * @return 
  26.      */ 
  27.     @Override 
  28.     public boolean isValid(Integer value, ConstraintValidatorContext context) { 
  29.         //判斷是否包含這個(gè)值 
  30.         return ints.contains(value); 
  31.     } 

如果約束注解需要對(duì)其他數(shù)據(jù)類型進(jìn)行校驗(yàn),則可以的自定義對(duì)應(yīng)數(shù)據(jù)類型的校驗(yàn)器,然后在約束注解頭上的@Constraint注解中指定其他的校驗(yàn)器。

演示

校驗(yàn)注解和校驗(yàn)器自定義成功之后即可使用,如下:

  1. @Data 
  2. public class AuthorDTO { 
  3.     @EnumValues(values = {1,2},message = "性別只能傳入1或者2"
  4.     private Integer gender; 

總結(jié) 

數(shù)據(jù)校驗(yàn)作為客戶端和服務(wù)端的一道屏障,有著重要的作用,通過這篇文章希望能夠?qū)SR-303數(shù)據(jù)校驗(yàn)有著全面的認(rèn)識(shí)。

 

責(zé)任編輯:武曉燕 來源: 碼猿技術(shù)專欄
相關(guān)推薦

2025-04-08 00:33:00

數(shù)據(jù)校驗(yàn)Bean

2021-12-23 11:10:38

稅收大數(shù)據(jù)大數(shù)據(jù)稅收

2021-03-26 05:59:10

內(nèi)存檢測(cè)工具

2025-06-03 17:37:49

模型訓(xùn)練AI

2020-03-06 08:15:54

新人技術(shù)主管陳琦

2022-06-30 12:44:48

黑客數(shù)據(jù)泄露

2015-05-18 09:23:04

JavaScript數(shù)組

2017-05-24 17:25:44

2022-12-31 13:53:53

工程

2018-11-07 12:19:18

辭退北大能力

2021-06-02 10:06:52

神經(jīng)網(wǎng)絡(luò)數(shù)據(jù)圖形

2025-04-23 08:02:44

2024-12-17 07:00:00

2021-02-07 18:19:44

RabbitMQ客戶端

2015-11-11 14:38:18

2024-10-17 09:45:03

2023-07-17 09:19:20

CSSCSS 漸變

2021-03-08 08:02:40

IDEA插件JSON

2020-09-08 08:45:39

jupyter插件代碼

2021-03-26 15:18:11

代碼工具Mockoon
點(diǎn)贊
收藏

51CTO技術(shù)棧公眾號(hào)

主站蜘蛛池模板: 久久免费电影 | 四虎av电影 | 色综合久久天天综合网 | 久热精品在线 | 九九热在线免费视频 | 在线观看免费毛片 | 福利视频一区 | 黄色一级在线播放 | 欧美精品在线一区二区三区 | 亚洲国产精品一区 | 亚洲一区二区三区四区在线观看 | 国产综合精品一区二区三区 | 在线国产小视频 | 成人国内精品久久久久一区 | 国产精品视频网址 | 亚洲国产精品第一区二区 | 国产高潮好爽受不了了夜色 | 精品国模一区二区三区欧美 | 欧美视频成人 | 狠狠ri| 免费网站国产 | 91精品国产91久久久久久不卞 | 亚洲国产高清高潮精品美女 | 91精品国产日韩91久久久久久 | 午夜丰满寂寞少妇精品 | 精品国产一区二区三区久久久蜜月 | 亚洲狠狠爱 | 国产伦一区二区三区久久 | 羞羞视频在线观免费观看 | 国产ts人妖一区二区三区 | 国产精品三级久久久久久电影 | 日韩精品中文字幕一区二区三区 | 亚洲欧美精品 | 日韩一区二区三区在线 | 亚洲黄色片免费观看 | 国产精品福利在线 | 亚洲成人网在线观看 | 欧美成人精品在线观看 | 日韩成人在线观看 | 国产综合精品一区二区三区 | 国产亚洲精品美女久久久久久久久久 |