JSR303参数校验-SpringMVC

news2024/9/22 21:16:39

文章目录

  • JSR303技术标准简介
  • JSR303标准几个具体实现框架
    • validation-api
    • jakarta.validation-api
    • hibernate-validator
    • spring-boot-starter-validation
  • Spring Validation
  • javax.validation.constraints包下提供的注解
  • org.hibernate.validator.constraints包扩展的注解
  • 校验注解默认错误信息
  • @Valid、 @Validated
  • SpringBoot-JSR-303数据校验
    • @RequestParam/@PathVariable传参
    • @RequestBody参数校验
    • 级联属性数据校验
    • 分组校验
    • 自定义校验注解
    • 组合多个校验注解成自定义注解
  • 统一校验异常处理
    • ConstraintViolationException.class
    • MethodArgumentNotValidException.class
    • BindException.class
  • 注解@Pattern 常用校验
    • 正则基本语法符号
    • 使用小技巧
    • 常用的正则校验
  • 枚举类型参数校验
    • User对象标注@EnumValue
    • 自定义@EnumValue注解(注解名随缘)
    • 枚举类提供逻辑校验的方法
    • 实现@EnumValue的校验器
    • 总结使用步骤
  • @RequestBody参数校验实现原理
  • RequestParam/PathVariable参数校验实现原理
  • 基于XML的配置校验器

JSR303技术标准简介

  1. JCP(Java Community Process)为Java技术制定标准技术规范的机构,任何人都可以注册 JCP 网站,并且可以参与JSR(Java Specification Requests:Java规范提案)的评审,也可以提交自己的JSR
  2. Bean Validation是Java定义的一套基于注解的数据校验规范,出自JSR303,JSR349,JSR380规范提案.

JSR303标准几个具体实现框架

JSR规范提案只是提供了技术标准,并没有提供具体的实现。具体相关框架有validation-apijakarta.validation-apihibernate-validator

validation-api

Java在2009年的Java EE 6中发布了JSR303以及javax.validation包,对JSR303版本的Bean Validation作了接口规范, 并没有相关校验代码。

<dependency>
    <groupId>javax.validation</groupId>
    <artifactId>validation-api</artifactId>
    <version>2.0.1.Final</version>
</dependency>

jakarta.validation-api

jakarta.validation-api是由validation-api改名而来。2018-03-05年,开源组织Eclipse基金会宣布将Java EE(Enterprise Edition:企业版)被更名为Jakarta EE(雅加达)。这是Oracle将Java EE移交给Eclipse基金会后加强对Java品牌控制的措施。随着Java EE的改名,包名和路径名也做了相应修改,从上面的引用路径可以看出来。

<dependency>
    <groupId>jakarta.validation</groupId>
    <artifactId>jakarta.validation-api</artifactId>
    <version>2.0.2</version>
</dependency>

hibernate-validator

hibernate-validator与持久层框架Hibernate没有什么关系,hibernate-validator是Hibernate基金会下的一个项目,是对Bean Validation的具体实现,它提供了JSR380规范中所有内置constraint的实现,还加入了它自己的一些constraint实现。点开pom文件,会发现hibernate-validator依赖于validation-api(jakarta.validation-api)。所以我们使用数据验证的时候可以直接引用这个包即可。

<dependency>
  	<groupId>org.hibernate.validator</groupId>
  	<artifactId>hibernate-validator</artifactId>
  	<version>6.1.6.Final</version>
</dependency>

spring-boot-starter-validation

对于SpringBoot项目,直接引用它提供的starter,这个starter内部依赖了hibernate-validator

<dependency>
    <groupId>org.springframework.boot</groupId>
    <artifactId>spring-boot-starter-validation</artifactId>
</dependency>

image.png

Spring Validation

Java API规范(JSR303)定义了Bean校验的标准validation-api,但没有提供实现。hibernate validation是对这个规范的实现,并增加了校验注解如@Email、@Length等。Spring Validation是对hibernate validation的二次封装,用于支持spring mvc参数自动校验

javax.validation.constraints包下提供的注解

注解说明可用类型
@AssertFalse必须为falseboolean,Boolean
@AssertTrue必须为trueboolean,Boolean
@DecimalMax(value=,inclusive=)必须是一个数字,其值必须小于或等于指定的最大值,支持的类型:BigDecimal、BigInteger、CharSequence、byte、short、int、long及其包装类BigDecimal,BigInteger,CharSequence,byte,short,int,long,Byte,Short,Integer,Long
@DecimalMin(value=,inclusive=)被注释的元素必须是一个数字,其值必须大于或等于指定的最小值。null 值是有效的。
value:用 String 表示 BigDecimal 字符串的最小值。
inclusive:true 表示大于或等于指定的最小值,false 表示大于最小值。默认为 true。
BigDecimal,BigInteger,CharSequence,byte,short,int,long,Byte,Short,Integer,Long
@Digits(integer=, fraction=)带注释的元素必须是可接受范围内的数字。null 值是有效的。
integer:此数字所接受的最大整数的位数。
fraction:该数字所接受的最大小数的位数。
BigDecimal,BigInteger,CharSequence,byte,short,int,long,Byte,Short,Integer,Long
@Email(regexp=, flags={})检查指定的字符序列是否为有效的电子邮件地址。允许通过可选参数regexp和flags指定电子邮件必须匹配的附加正则表达式(包括正则表达式标志)。null 值是有效的。CharSequence
@Future被注释的元素必须是一个未来的日期时间。null 值是有效的。和@Past相同
@FutureOrPresent被注释的元素必须是一个现在或未来的日期时间。null 值是有效的。和@Past相同
@Max(value=)被注释的元素必须是一个数字,其值必须小于或等于指定的最大值。null 值是有效的。
value:指定元素必须小于或等于的 long 值。
BigDecimal,BigInteger,byte,short,int,long,Byte,Short,Integer,Long
@Min(value=)被注释的元素必须是一个数字,其值必须大于或等于指定的最小值。null 值是有效的。
value:指定元素必须大于或等于的 long 值。
BigDecimal,BigInteger,byte,short,int,long,Byte,Short,Integer,Long
@Negative必须是严格的负数(即0被认为是无效值), null 值是有效的。BigDecimal,BigInteger,byte,short,int,long,float,double,Byte,Short,Integer,Long,Float,Double
@NegativeOrZero被注释的元素必须是负数或 0。null 值是有效的。BigDecimal,BigInteger,byte,short,int,long,float,double,Byte,Short,Integer,Long,Float,Double
@NotBlank被注释的元素不能是 null,并且必须包含至少一个非空白字符。CharSequence
@NotEmpty被注释的元素不能是 null 或空。CharSequence:计算字符序列的长度。
Collection:计算集合大小。
Map:计算 map 的大小。
Array:计算数组长度。
@NotNull不能为null任何类型
@Null必须为null任何类型
@Past被注释的元素必须是一个过去的日期时间。null 值是有效的。java.util.Date
java.util.Calendar
java.time.Instant
java.time.LocalDate
java.time.LocalDateTime
java.time.LocalTime
java.time.MonthDay
java.time.OffsetDateTime
java.time.OffsetTime
java.time.Year
java.time.YearMonth
java.time.ZonedDateTime
java.time.chrono.HijrahDate
java.time.chrono.JapaneseDate
java.time.chrono.MinguoDate
java.time.chrono.ThaiBuddhistDate
@PastOrPresent被注释的元素必须是一个过去或现在的日期时间。null 值是有效的。和@Past相同
@Pattern(regexp=, flags={})被注释的元素必须匹配指定的正则表达式。正则表达式遵循 Java 正则表达式约定,见 Java.util.regex.Pattern。null 值是有效的。
regexp:要匹配的正则表达式。
flags:解析正则表达式时考虑的 Flag 枚举数组。
CharSequence
@Positive被注释的元素必须是一个严格的正数(即 0 被认为是无效的值)。null 值是有效的。BigDecimal,BigInteger,byte,short,int,long,float,double,Byte,Short,Integer,Long,Float,Double
@PositiveOrZero被注释的元素必须是正数或 0。null 值是有效的。BigDecimal,BigInteger,byte,short,int,long,float,double,Byte,Short,Integer,Long,Float,Double
@Size(min=, max=)被注释的元素大小必须介于最小和最大(闭区间)之间。null 值是有效的。
min:指定元素必须大于或等于的 int 值,默认为 0。
max:指定元素必须小于或等于的 int 值,默认为 int 类型的最大值。
CharSequence:计算字符序列的长度。
Collection:计算集合大小。
Map:计算 map 的大小。
Array:计算数组长度。
@URL校验URL的合法性,但是并不完善,例如:https://;https:www.baidu.com等校验不住借助它regex属性,指定正则表达式进行校验
^(http|https)😕/[a-zA-Z0-9]+([\\-\\.]{1}[a-zA-Z0-9]+)\\.[a-zA-Z]{2,5}(:[0-9]{1,5})?(\\/.)?$

org.hibernate.validator.constraints包扩展的注解

  • 具体的注解可到org.hibernate.validator.constraints包下面自行查看
  • 校验实现可以到org.hibernate.validator.internal.constraintvalidators包下面自行查看。
    • 其中bv包下是对javax.validation下注解的实现,
    • hv包下是对org.hibernate.validator下注解的实现
注解描述
URL字符串是否为URL
CNPJ验证CNPJ(巴西企业纳税人注册号)
CPF验证CPF(巴西个人纳税人登记号)
TituloEleitoral巴西选民身份证号码
NIP验证是否为NIP编号(9位波兰增值税标识号)
PESEL验证PESEL(波兰国家识别号)
INN俄罗斯纳税人识别号
REGON验证REGON编号(9/14位波兰纳税人识别号)
DurationMax必须小于或等于DurationMax#nanos()、DurationMax#millis()等
DurationMin必须大于于或等于DurationMax#nanos()、DurationMax#millis()等
CodePointLength验证字符序列的长度是否在最小值和最大值之间
ConstraintComposition应用于组合约束注释的所有约束的布尔运算符
CreditCardNumber带注释的元素必须表示有效的信用卡号码,这是Luhn算法的实现,它的目的是检查用户的错误,而不是信用卡的有效性
Currency必须是正确的
EAN列是否为有效数字,并验证校验位
ISBN验证ISBN字符是否有效,号码的长度和校验位都经过验证
Length验证字符串是否在最小值和最大值之间
LuhnCheckLuhn算法检查约束
Mod10Check允许验证一系列数字是否通过Mod10校验和算法。经典的Mod10是通过对每个奇数的数字求和来计算的数字(从右到左)值乘以{@code multiplier}。
Mod11Check验证一系列数字是否通过Mod11校验和算法。对于最常见的Mod11变量,总和计算是通过将从最右边的数字(不包括校验数字)到最左边的数字。重量从2开始,每个数字增加1。然后将结果用于使用{@code11-(sum%11)}计算校验位
Normalized验证字符序列是否为规范化形式。可以通过设置规范化策略来验证规范化值
ParameterScriptAssert方法级约束,用于根据带注释的方法或构造函数。此约束可用于实现依赖于带注释对象的多个参数的验证例程
Range必须在适当的范围内。应用于数值或字符串数值的表示
ScriptAssert类级约束,用于根据带注释的元素。此约束可用于实现验证依赖于带注释元素的多个属性的例程
UniqueElements验证所提供的{@link Collection}中的每个对象都是唯一的,即我们在其中找不到2个相等的元素

校验注解默认错误信息

每个注解都有默认错误提示,如果没有写message属性的话则默认显示文件中配置的提示信息,我们用的是简体中文的ValidationMessages_zh_CN.properties文件,文件路径:

hibernate-validator-6.1.6.Final.jar!\org\hibernate\validator\ValidationMessages_zh_CN.properties

默认提示息如下所示:

javax.validation.constraints.AssertFalse.message     = 必须为 false
javax.validation.constraints.AssertTrue.message      = 必须为 true
javax.validation.constraints.DecimalMax.message      = 必须小于 ${inclusive == true ? 'or equal to ' : ''}{value}
javax.validation.constraints.DecimalMin.message      = 必须大于 ${inclusive == true ? 'or equal to ' : ''}{value}
javax.validation.constraints.Digits.message          = 数字值超出了边界(期望 <{integer} digits>.<{fraction} digits>)
javax.validation.constraints.Email.message           = 必须为格式规范的电子邮件地址
javax.validation.constraints.Future.message          = 必须是未来的日期
javax.validation.constraints.FutureOrPresent.message = 必须是现在或将来的日期
javax.validation.constraints.Max.message             = 必须小于或等于 {value}
javax.validation.constraints.Min.message             = 必须大于或等于 {value}
javax.validation.constraints.Negative.message        = 必须小于 0
javax.validation.constraints.NegativeOrZero.message  = 必须小于或等于 0
javax.validation.constraints.NotBlank.message        = 不得为空白
javax.validation.constraints.NotEmpty.message        = 不得为空
javax.validation.constraints.NotNull.message         = 不得为 null
javax.validation.constraints.Null.message            = 必须为 null
javax.validation.constraints.Past.message            = 必须是过去的日期
javax.validation.constraints.PastOrPresent.message   = 必须是过去或现在的日期
javax.validation.constraints.Pattern.message         = 必须与 "{regexp}" 匹配
javax.validation.constraints.Positive.message        = 必须大于 0
javax.validation.constraints.PositiveOrZero.message  = 必须大于或等于 0
javax.validation.constraints.Size.message            = 大小必须在 {min} 和 {max} 之间

org.hibernate.validator.constraints.CreditCardNumber.message        = 无效信用卡卡号
org.hibernate.validator.constraints.Currency.message                = 无效货币(必须为 {value} 之一)
org.hibernate.validator.constraints.EAN.message                     = 无效 {type} 条形码
org.hibernate.validator.constraints.Email.message                   = 电子邮件地址格式不规范
org.hibernate.validator.constraints.ISBN.message                    = 无效 ISBN
org.hibernate.validator.constraints.Length.message                  = 长度必须介于 {min} 与 {max} 之间
org.hibernate.validator.constraints.CodePointLength.message         = 长度必须介于 {min} 与 {max} 之间
org.hibernate.validator.constraints.LuhnCheck.message               = ${validatedValue} 的校验位无效,Luhn Modulo 10 校验和失败
org.hibernate.validator.constraints.Mod10Check.message              = ${validatedValue} 的校验位无效,Modulo 10 校验和失败
org.hibernate.validator.constraints.Mod11Check.message              = ${validatedValue} 的校验位无效,Modulo 11 校验和失败
org.hibernate.validator.constraints.ModCheck.message                = ${validatedValue} 的校验位无效,{modType} 校验和失败
org.hibernate.validator.constraints.NotBlank.message                = 可能不为空
org.hibernate.validator.constraints.NotEmpty.message                = 可能不为空
org.hibernate.validator.constraints.ParametersScriptAssert.message  = 脚本表达式 "{script}" 未求值为 true
org.hibernate.validator.constraints.Range.message                   = 必须介于 {min} 与 {max} 之间
org.hibernate.validator.constraints.SafeHtml.message                = 可能具有不安全的 HTML 内容
org.hibernate.validator.constraints.ScriptAssert.message            = 脚本表达式 "{script}" 未求值为 true
org.hibernate.validator.constraints.UniqueElements.message          = 必须仅包含唯一元素
org.hibernate.validator.constraints.URL.message                     = 必须为有效 URL

org.hibernate.validator.constraints.br.CNPJ.message                 = 无效巴西企业纳税人登记号 (CNPJ)
org.hibernate.validator.constraints.br.CPF.message                  = 无效巴西个人纳税人登记号 (CPF)
org.hibernate.validator.constraints.br.TituloEleitoral.message      = 无效巴西投票人身份证号

org.hibernate.validator.constraints.pl.REGON.message                = 无效波兰纳税人识别号 (REGON)
org.hibernate.validator.constraints.pl.NIP.message                  = 无效 VAT 识别号 (NIP)
org.hibernate.validator.constraints.pl.PESEL.message                = 无效波兰身份证号 (PESEL)

org.hibernate.validator.constraints.time.DurationMax.message        = 必须短于 ${inclusive == true ? ' or equal to' : ''}${days == 0 ? '' : days == 1 ? ' 1 day' : ' ' += days += ' days'}${hours == 0 ? '' : hours == 1 ? ' 1 hour' : ' ' += hours += ' hours'}${minutes == 0 ? '' : minutes == 1 ? ' 1 minute' : ' ' += minutes += ' minutes'}${seconds == 0 ? '' : seconds == 1 ? ' 1 second' : ' ' += seconds += ' seconds'}${millis == 0 ? '' : millis == 1 ? ' 1 milli' : ' ' += millis += ' millis'}${nanos == 0 ? '' : nanos == 1 ? ' 1 nano' : ' ' += nanos += ' nanos'}
org.hibernate.validator.constraints.time.DurationMin.message        = 必须长于 ${inclusive == true ? ' or equal to' : ''}${days == 0 ? '' : days == 1 ? ' 1 day' : ' ' += days += ' days'}${hours == 0 ? '' : hours == 1 ? ' 1 hour' : ' ' += hours += ' hours'}${minutes == 0 ? '' : minutes == 1 ? ' 1 minute' : ' ' += minutes += ' minutes'}${seconds == 0 ? '' : seconds == 1 ? ' 1 second' : ' ' += seconds += ' seconds'}${millis == 0 ? '' : millis == 1 ? ' 1 milli' : ' ' += millis += ' millis'}${nanos == 0 ? '' : nanos == 1 ? ' 1 nano' : ' ' += nanos += ' nanos'}

@Valid、 @Validated

注解完整类型描述
@Validjavax.validation.Valid标记用于验证级联的属性、方法参数或方法返回类型。在验证属性、方法参数或方法返回类型时,将验证在对象及其属性上定义的约束。
@Validatedorg.springframework.validation.annotation.ValidatedSpring 提供的扩展注解,可以添加在类、方法参数、普通方法上,表示它们需要进行约束校验,可以很方便的用于分组校验。

SpringBoot-JSR-303数据校验

@RequestParam/@PathVariable传参

GET请求一般会使用requestParam/PathVariable传参。如果参数比较多(比如超过6个),还是推荐使用DTO对象接收。否则,推荐将一个个参数平铺到方法入参中。在这种情况下,必须在Controller类上标注@Validated注解,并在入参上声明约束注解(如@Min等)。如果校验失败,会抛出ConstraintViolationException异常。代码示例如下

@RestController
@Validated
public class GroupController {

	@PostMapping("/audit/flow")
    public String startAuditFlow(@NotBlank(message = "申请ID不能为空") String applyRecordId) {
        // TODO
    }
    
}


@RequestBody参数校验

  1. POST、PUT请求一般会使用@RequestBody传递参数,这种情况下,后端使用VO对象进行接收。只要给VO对象加上@Validated注解就能实现自动参数校验。比如,有一个保存User的接口,要求userName长度是2-10,account和password字段长度是6-20。
  2. 如果校验失败,会抛出MethodArgumentNotValidException异常,Spring默认会将其转为400(Bad Request)请求。
@Data
public class UserDTO {

    private Long userId;

    @NotNull
    @Length(min = 2, max = 10)
    private String userName;

    @NotNull
    @Length(min = 6, max = 20)
    private String account;

    @NotNull
    @Length(min = 6, max = 20)
    private String password;
}


在方法参数上声明校验注解, 这种情况下,使用@Valid和@Validated都可以

@PostMapping("/save")
public Result saveUser(@RequestBody @Validated UserDTO userDTO) {
    // 校验通过,才会执行业务逻辑处理
    return Result.ok();
}

级联属性数据校验

1、如果User对象的级联属性address也需要校验; 只需在对应的address属性上添加@Valid注解即可,如下的代码片段

import lombok.Data;

import javax.validation.Valid;
import javax.validation.constraints.NotEmpty;

@Data
public class User {

    @NotEmpty(message = "人员姓名不能为空")
    private String name;
    
    private String id;

    // 注意 @Valid 注解的标注位置
    private @Valid Address address;
}


// ===================================级联的地址对象================================

import lombok.Data;

import javax.validation.constraints.NotEmpty;

@Data
public class Address {

    @NotEmpty(message = "住址所在省份名称不能为空")
    private String province;
}

2、 使用@Validated或者@Valid注解开启SpringMVC在进行数据绑定的同时进行数据校验

@RestController
public class UserController {

    @PostMapping("/user/create")
    public Result<Object> createUser(@RequestBody @Validated User user) {
        return Result.of().setCode("2000").setMessage("成功").setData(user);
    }
}

分组校验

不同场景下的不同请求使用同一个Model对象接受请求参数时,存在不同的请求需要对Model对象的同一个属性制定不同的校验逻辑;比如上面的User对象的id属性;创建User对象的时候id是非必传;但是在更新User对象时id属性是必需的

1、 单纯的声明两个接口,充当分组

/**
 * @User 请求模型参数校验分组:创建用户时指定
 */
interface CreateUserValidGroup {
    // 不需要任何定义
}

/**
 * @User 请求模型参数校验分组:更新用户时指定
 */
interface UpdateUserValidGroup {
    // 不需要任何的定义
}

2、对标注的字段进行分组标记,使用校验注解的groups属性

@Data
public class User {

    @NotEmpty(message = "人员姓名不能为空")
    private String name;

    /**
     * 分组校验:创建用户的时候id是空的;用户信息更新的时候id是必录的
     * @Blank是自定义校验注解
     */
    @NotEmpty(message = "用户Id不能为空,在用户进行数据更新的时候",groups = {UpdateUserValidGroup.class})
    @Blank(message = "用户Id必须为空,在创建用户的时候",groups = {CreateUserValidGroup.class})
    private String id;

    private @Valid Address address;
}

3、使用@Validated告诉SpringMVC使用哪组校验进行数据校验

@RestController
public class UserController {

    @PostMapping("/user/create")
    public Result<Object> createUser(@RequestBody @Validated({CreateUserValidGroup.class}) User user) {
        return Result.of().setCode("2000").setMessage("成功").setData(user);
    }

    @PostMapping("/user/update")
    public Result<Object> modifyUser(@RequestBody @Validated({UpdateUserValidGroup.class}) User user) {
        return Result.of().setCode("2000").setMessage("成功").setData(user);
    }
}

自定义校验注解

有时候框架提供的约束注解并不能满足我们的需求,所以需要自定义合适的校验规则来满足自己的校验需求。

自定义注解用到的注解:

  • @Target:指定此注解支持的元素类型,比如:FIELD(属性)、METHOD(方法)等;
    
  • @Rentention(RUNTIME):指定此类型的注解将在运行时通过反射方式使用;
    
  • @Constraint(validatedBy = { }):标记注解的类型为约束,指定注解所使用的验证器(写验证逻辑的类),如果约束可以用在多种数据类型中,则每种数据类型对应一个验证器。
    
  • @Documented:表示带有某种类型的注释将由 javadoc 和类似工具记录;
    
  • @Repeatable(List.class):指示注解可以在相同的位置重复多次,通常具有不同的配置。List 包含注解类型。
    

Bean Validation API 规范要求任何约束注解都需要定义以下属性:

  1. message:定制化的提示信息,主要是从ValidationMessages.properties里提取,也可以依据实际情况进行定制。
    
  2. groups:用于将约束注解进行分组,不同组会执行不同的校验操作。
    
  3. payload:主要是针对Bean的,使用不多,未做详细了解。
    

为每一个自定义注解添加指定的校验器

@Documented
@Constraint(
    	// 指定解析该注解的类
        validatedBy = {BlankValidator.class}
)
@Target({ElementType.METHOD, ElementType.FIELD, ElementType.ANNOTATION_TYPE, ElementType.CONSTRUCTOR, ElementType.PARAMETER, ElementType.TYPE_USE})
@Retention(RetentionPolicy.RUNTIME)
public @interface Blank {
    String message() default "{com.idooy.param.constraint.message}";

    Class<?>[] groups() default {};

    Class<? extends Payload>[] payload() default {};
}

对自定义的校验注解进行解析的类BlankValidator.class,需要实现特定的接口 ConstraintValidator

import org.springframework.util.StringUtils;

import javax.validation.ConstraintValidator;
import javax.validation.ConstraintValidatorContext;

public class BlankValidator implements ConstraintValidator<Blank,String> {

    @Override
    public void initialize(Blank constraintAnnotation) {
        ConstraintValidator.super.initialize(constraintAnnotation);
    }

    @Override
    public boolean isValid(String value, ConstraintValidatorContext context) {
        return !StringUtils.hasText(value);
    }
}

组合多个校验注解成自定义注解

@Min(value = 18, message = "年龄最小不能小于18")
@Max(value = 120, message = "年龄最大不能超过120")
@Constraint(validatedBy = {})//不指定校验器
@Documented
@Target({ElementType.ANNOTATION_TYPE, ElementType.METHOD, ElementType.FIELD})
@Retention(RetentionPolicy.RUNTIME)
public @interface CustomAgeValidator {
    String message() default "年龄大小必须大于18并且小于120";

    Class<?>[] groups() default {};

    Class<? extends Payload>[] payload() default {};
}

在创建完自定义注解后,我们在User类的age属性上使用自定义组合注解。

public class User {
    @NotBlank(message = "姓名不允许为空")
    @Length(min = 2, max = 10, message = "姓名长度错误,长度长度2-10!")
    private String name;

//    @NotNull(message = "年龄不能为空")
//    @Min(value = 18,message = "年齡最小不能小于18")
    @CustomAgeValidator
    private int age;

    @NotBlank(message = "地址不能为空")
    private String address;

    @Pattern(regexp = "^1(3[0-9]|5[012356789]|7[1235678]|8[0-9])\\d{8}$", message = "手机号格式错误")
    private String phone;

    @Email(message = "邮箱格式错误")
    private String email;

}

统一校验异常处理

ConstraintViolationException.class

抛出的场景:普通参数(非 java bean)校验出错时,会抛出 ConstraintViolationException 异常 比如:getUser(@NotBlank(message = "id不能为空") String id),GET请求一般会使用requestParam/PathVariable传参 ,将一个个参数平铺到方法入参中。在这种情况下,必须在Controller类上标注@Validated注解,并在入参上声明约束注解(如@Min等)。如果校验失败,会抛出ConstraintViolationException异常

@RequestMapping("/api/user")
@RestController
@Validated
public class UserController {
    // 路径变量
    @GetMapping("{userId}")
    public Result detail(@PathVariable("userId") @Min(10000000000000000L) Long userId) {
        // 校验通过,才会执行业务逻辑处理
        return Result.ok(userDTO);
    }

    // 查询参数
    @GetMapping("getByAccount")
    public Result getByAccount(@Length(min = 6, max = 20) @NotNull String  account) {
        // 校验通过,才会执行业务逻辑处理
        return Result.ok(userDTO);
    }
}
    /**
     * JSR303校验异常
     * 普通参数(非 java bean)校验出错时,会抛出 ConstraintViolationException 异常
     * @param ex
     * @return
     */
    @ExceptionHandler(ConstraintViolationException.class)
    public R constraintViolationException(ConstraintViolationException ex){
        Set<ConstraintViolation<?>> violations = ex.getConstraintViolations();
        String message = violations.stream().map(ConstraintViolation::getMessage).collect(Collectors.joining(";"));
        return R.error(message);
    }

MethodArgumentNotValidException.class

抛出的场景:RestFul风格下POST、PUT请求一般会使用JSON格式的请求体绑定到VO对象上,并且使用@RequestBody注解。这种情况下校验参数失败时抛出MethodArgumentNotValidException

@PostMapping("/save")
public Result saveUser(@RequestBody @Validated UserDTO userDTO) {
    // 校验通过,才会执行业务逻辑处理
    return Result.ok();
}
    /**
     * JSR303校验异常
     * RestFul风格下POST、PUT请求一般会使用JSON格式的请求体绑定到VO对象上,
     * 并且使用@RequestBody注解。这种情况下校验参数失败时抛出MethodArgumentNotValidException
     * @param e
     * @return
     */
    @ExceptionHandler(MethodArgumentNotValidException.class)
    public R methodArgumentNotValidExceptionHandler(MethodArgumentNotValidException e) {
        BindingResult bindingResult = e.getBindingResult();
        FieldError fieldError = bindingResult.getFieldError();
        if (fieldError != null) {
            return R.error(fieldError.getField() + " : " + fieldError.getDefaultMessage());
        }
        // 如果校验注解标注在类上,比如@PasswordMatches;此时FieldError就一定为null;
        // 所以可以这样获取错误信息
        String err = bindingResult.getAllErrors().get(0).getDefaultMessage();
        return R.error(err);
    }

BindException.class

抛出的场景: 入参绑定到javaBean(json格式,或者表单提交都是可以绑定到VO对象上),但是没有标注@RequestBody注解,示例代码如下:

@RestController
    @RequestMapping("/member")
    public class MemberController {

        @PostMapping("/build")
        public Result buildMember(@Valid Member member) {
            return Result.of().setData(member);
        }
    }

统一的异常处理代码片段

        /**
     * JSR303校验异常
     * 抛出的场景: 入参绑定到javaBean(json格式,或者表单提交都是可以绑定到VO对象上),
     * 但是没有标注@RequestBody注解,示例代码如下
     * @param ex
     * @return
     */
    @ExceptionHandler(BindException.class)
    public R bindExceptionHandle(BindException ex){
        ObjectError objectError = ex.getAllErrors().get(0);
        String defaultMessage = objectError.getDefaultMessage();
        return R.error(defaultMessage);
    }

注解@Pattern 常用校验

正则基本语法符号

首先来从基本语法符号开始:

  • ^ 表示匹配字符串的开始位置 (例外 用在中括号中[ ] 时,可以理解为取反,表示不匹配括号中字符串)
  • $ 表示匹配字符串的结束位置
  • * 表示匹配 零次到多次
  • + 表示匹配 一次到多次 (至少有一次)
  • ? 表示匹配零次或一次
  • . (圆点)表示匹配单个字符
  • | 表示为或者,两项中取一项(可用于两个正则之间)
  • ( ) 小括号表示匹配括号中全部字符
  • [ ] 中括号表示匹配括号中一个字符 范围描述 如[0-9 a-z A-Z]
  • { } 大括号用于限定匹配次数 如 {n}表示匹配n个字符 {n,}表示至少匹配n个字符 {n,m}表示至少n,最多m
  • \ 转义字符 如上基本符号匹配都需要转义字符 如 *表示匹配*号
  • \w 表示英文字母和数字 \W 非字母和数字
  • \d 表示数字 \D 非数字
  • \s表示空字符串
  • 当使用-时,需要放在最后一个位置,否则会被当成数字和字母之间区间的标识

使用小技巧

  1. 首先是首尾匹配,那么就要使用^和$,使用*或者\的时候,就要使用转义字符了(下面是在java中使用的,转义符号需要两个斜杠配合使用才好使)。
  2. 然后是长度限制时,可以使用大括号{ } {n}表示匹配n个字符 {n,}表示至少匹配n个字符 {n,m}表示至少n,最多m

常用的正则校验

手机号:
"^1(3[0-9]|5[012356789]|7[1235678]|8[0-9])\\d{8}$"

枚举类型参数校验

接口中某个常量参数值,需要映射业务枚举类中的某个实例,比如,审核状态枚举类定义了四个实例 “0”,“待审核”、“1”,“审核中”、“2”,“审核通过”、“3”,“审核驳回”;我们期望将参数中的常量进行验证并转换成对应的枚举实例。
思路

该功能仅仅满足如下需求:请求参数的常量值依赖枚举类型进行验证是否合法;并不能将请求参数常量直接映射成对应的枚举实例。如果期望‘验证+映射’功能可参看接下来的另一篇文章

  1. VO对象待校验的常量字段类型就用String或者Integer类型接。因为直接使用具体的枚举类型接就是类型转换器或者jackson的反序列化操作,难度较大(另起一文)。
  2. 枚举类中定义方法自行判断常量值是否可以映射具体的枚举实例。
  3. 注解校验器中使用反射调用第二步提供的方法。

具体的实现过程如下;

controllerc层Handler入参还是使用User;所以其中分组校验功能可以不用关注

    @PostMapping("/user/update")
    public Result<Object> modifyUser(@RequestBody @Validated({UpdateUserValidGroup.class}) User user) {
        
    }

User对象标注@EnumValue

其中的approvalStatus就是此次校验特别关注的字段

@Data
public class User {

    @NotEmpty(message = "人员姓名不能为空")
    private String name;

    /**
     * 分组校验:创建用户的时候id是空的;用户信息更新的时候id是必录的
     */
    @NotEmpty(message = "用户Id不能为空,在用户进行数据更新的时候",groups = {UpdateUserValidGroup.class})
    @Blank(message = "用户Id必须为空,在创建用户的时候",groups = {CreateUserValidGroup.class})
    private String id;

    @EnumValue(enumClass = ApprovalStatusEnums.class,groups = {UpdateUserValidGroup.class, CreateUserValidGroup.class})
    private String approvalStatus;
    private @Valid Address address;
}

接下来就是EnumValue注解类的定义,以及它所对应的校验器类EnumValueValidator

自定义@EnumValue注解(注解名随缘)

package com.idooy.param;

import javax.validation.Constraint;
import javax.validation.Payload;
import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;

/**
 * 校验当前被标注的字段传入的值是否可以正确映射为指定的枚举类型
 *
 * @author duheng
 */
@Target({ElementType.METHOD, ElementType.FIELD, ElementType.ANNOTATION_TYPE, ElementType.PARAMETER})
@Retention(RetentionPolicy.RUNTIME)
@Constraint(validatedBy = EnumValueValidator.class)
public @interface EnumValue {

    String message() default "com.idooy.param: invalid enum type value";

    Class<?>[] groups() default {};

    Class<? extends Payload>[] payload() default {};

    /**
     * 当前被校验的字段值,要和哪个枚举类型进行映射
     * @return
     */
    Class<? extends Enum<?>> enumClass();

    /**
     * 指定枚举类中的一个存在的方法名:验证参数是否符合要求
     * @return
     */
    String enumMethod() default "isValidEnum";
}

枚举类提供逻辑校验的方法

枚举中需要提供一个可供业务逻辑判断的方法,@EnumValue默认执行isValidEnum方法。

package com.idooy.param;

import lombok.AllArgsConstructor;
import lombok.Getter;

import java.util.Arrays;
import java.util.Objects;

/**
 * @author duheng
 */
@AllArgsConstructor
@Getter
public enum ApprovalStatusEnums {

    /**
     * 待审核
     */
    TO_BE_REVIEW("0","待审核"),
    /**
     * 审核中
     */
    REVIEWING("1","审核中"),

    /**
     * 业务员审核通过--->用户加入自定义群组的申请
     */
    REVIEW_AGREE("2","审核通过"),

    /**
     * 业务员审核驳回--->用户加入自定义群组的申请
     */
    REVIEW_REJECT("3","审核驳回");

    private String value;

    private String message;


    /**
     * 获取当前枚举的有效值
     * @param code
     * @return
     */
    public static boolean isValidEnum(String code) {
        return Arrays.stream(ApprovalStatusEnums.values())
                .anyMatch(o -> Objects.equals(o.getValue(), code));
    }
}

如果不使用默认的方法名isValidEnum,就可以在标注@EnumValue时指定方法名

实现@EnumValue的校验器

package com.idooy.param;

import org.springframework.util.StringUtils;

import javax.validation.ConstraintValidator;
import javax.validation.ConstraintValidatorContext;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
import java.lang.reflect.Modifier;

/**
 *
 * @author duheng
 */
public class EnumValueValidator implements ConstraintValidator<EnumValue, Object> {
    private Class<? extends Enum<?>> enumClass;
    private String enumMethod;

    @Override
    public void initialize(EnumValue enumValue) {
        enumMethod = enumValue.enumMethod();
        enumClass = enumValue.enumClass();
    }

    /**
     *
     * @param value
     * @param constraintValidatorContext
     * @return
     */
    @Override
    public boolean isValid(Object value, ConstraintValidatorContext constraintValidatorContext) {
        // 对 null 或者 空字符串 不生效
        if (value == null) {
            return Boolean.TRUE;
        }
        if (value instanceof String && !StringUtils.hasText(value.toString())) {
            return Boolean.TRUE;
        }

        Class<?> valueClass = value.getClass();
        try {
            Method method = enumClass.getMethod(enumMethod, valueClass);
            if (!Boolean.TYPE.equals(method.getReturnType()) && !Boolean.class.equals(method.getReturnType())) {
                throw new RuntimeException(
                        String.format("%s method return is not boolean type in the %s class", enumMethod, enumClass));
            }

            if (!Modifier.isStatic(method.getModifiers())) {
                throw new RuntimeException(
                        String.format("%s method is not static method in the %s class", enumMethod, enumClass));
            }

            Boolean result = (Boolean) method.invoke(null, value.toString());
            return result == null ? false : result;
        } catch (IllegalAccessException | IllegalArgumentException | InvocationTargetException e) {
            throw new RuntimeException(e);
        } catch (NoSuchMethodException | SecurityException e) {
            throw new RuntimeException(
                    String.format("This %s(%s) method does not exist in the %s", enumMethod, valueClass, enumClass), e);
        }
    }
}

总结使用步骤

如上;在完成EnumValue.javaEnumValueValidator.java编写后,该如何使用自定义注解@EnumValue完成String类型到枚举类型映射校验呢?

  1. @EnumValue对null和空串不生效;约定,入参最好是String或者Integer
  2. 保证枚举类中存在方法isValidEnum,要求:公共、静态、返回值为Boolean、入参有且只有一个String类型;也可自定义方法名,在使用@EnumValue时使用enumMethod指定
  3. 标注使用@EnumValue(enumClass = ApprovalStatusEnums.class)

@RequestBody参数校验实现原理

在spring-mvc中,RequestResponseBodyMethodProcessor是用于解析@RequestBody标注的参数以及处理@ResponseBody标注方法的返回值的。显然,执行参数校验的逻辑肯定就在解析参数的方法resolveArgument()中:

public class RequestResponseBodyMethodProcessor extends AbstractMessageConverterMethodProcessor {
    @Override
    public Object resolveArgument(MethodParameter parameter, @Nullable ModelAndViewContainer mavContainer,
                                  NativeWebRequest webRequest, @Nullable WebDataBinderFactory binderFactory) throws Exception {

        parameter = parameter.nestedIfOptional();
        //将请求数据封装到DTO对象中
        Object arg = readWithMessageConverters(webRequest, parameter, parameter.getNestedGenericParameterType());
        String name = Conventions.getVariableNameForParameter(parameter);

        if (binderFactory != null) {
            WebDataBinder binder = binderFactory.createBinder(webRequest, arg, name);
            if (arg != null) {
                // 执行数据校验
                validateIfApplicable(binder, parameter);
                if (binder.getBindingResult().hasErrors() && isBindExceptionRequired(binder, parameter)) {
                    throw new MethodArgumentNotValidException(parameter, binder.getBindingResult());
                }
            }
            if (mavContainer != null) {
                mavContainer.addAttribute(BindingResult.MODEL_KEY_PREFIX + name, binder.getBindingResult());
            }
        }
        return adaptArgumentIfNecessary(arg, parameter);
    }
}

可以看到,resolveArgument()调用了validateIfApplicable()进行参数校验。

protected void validateIfApplicable(WebDataBinder binder, MethodParameter parameter) {
    // 获取参数注解,比如@RequestBody、@Valid、@Validated
    Annotation[] annotations = parameter.getParameterAnnotations();
    for (Annotation ann : annotations) {
        // 先尝试获取@Validated注解
        Validated validatedAnn = AnnotationUtils.getAnnotation(ann, Validated.class);
        //如果直接标注了@Validated,那么直接开启校验。
        //如果没有,那么判断参数前是否有Valid起头的注解。
        if (validatedAnn != null || ann.annotationType().getSimpleName().startsWith("Valid")) {
            Object hints = (validatedAnn != null ? validatedAnn.value() : AnnotationUtils.getValue(ann));
            Object[] validationHints = (hints instanceof Object[] ? (Object[]) hints : new Object[] {hints});
            //执行校验
            binder.validate(validationHints);
            break;
        }
    }
}

看到这里,大家应该能明白为什么这种场景下@Validated、@Valid两个注解可以混用。我们接下来继续看WebDataBinder.validate()实现。

@Override
public void validate(Object target, Errors errors, Object... validationHints) {
    if (this.targetValidator != null) {
        processConstraintViolations(
            //此处调用Hibernate Validator执行真正的校验
            this.targetValidator.validate(target, asValidationGroups(validationHints)), errors);
    }
}

最终发现底层最终还是调用了Hibernate Validator进行真正的校验处理。

RequestParam/PathVariable参数校验实现原理

上面提到的将参数一个个平铺到方法参数中,然后在每个参数前面声明约束注解的校验方式,就是方法级别的参数校验。实际上,这种方式可用于任何Spring Bean的方法上,比如Controller/Service等。其底层实现原理就是AOP,具体来说是通过MethodValidationPostProcessor动态注册AOP切面,然后使用MethodValidationInterceptor对切点方法织入增强

public class MethodValidationPostProcessor extends AbstractBeanFactoryAwareAdvisingPostProcessorimplements InitializingBean {
    @Override
    public void afterPropertiesSet() {
        //为所有`@Validated`标注的Bean创建切面
        Pointcut pointcut = new AnnotationMatchingPointcut(this.validatedAnnotationType, true);
        //创建Advisor进行增强
        this.advisor = new DefaultPointcutAdvisor(pointcut, createMethodValidationAdvice(this.validator));
    }

    //创建Advice,本质就是一个方法拦截器
    protected Advice createMethodValidationAdvice(@Nullable Validator validator) {
        return (validator != null ? new MethodValidationInterceptor(validator) : new MethodValidationInterceptor());
    }
}

接着看一下MethodValidationInterceptor

public class MethodValidationInterceptor implements MethodInterceptor {
    @Override
    public Object invoke(MethodInvocation invocation) throws Throwable {
        //无需增强的方法,直接跳过
        if (isFactoryBeanMetadataMethod(invocation.getMethod())) {
            return invocation.proceed();
        }
        //获取分组信息
        Class<?>[] groups = determineValidationGroups(invocation);
        ExecutableValidator execVal = this.validator.forExecutables();
        Method methodToValidate = invocation.getMethod();
        Set<ConstraintViolation<Object>> result;
        try {
            //方法入参校验,最终还是委托给Hibernate Validator来校验
            result = execVal.validateParameters(
                invocation.getThis(), methodToValidate, invocation.getArguments(), groups);
        }
        catch (IllegalArgumentException ex) {
            ...
        }
        //有异常直接抛出
        if (!result.isEmpty()) {
            throw new ConstraintViolationException(result);
        }
        //真正的方法调用
        Object returnValue = invocation.proceed();
        //对返回值做校验,最终还是委托给Hibernate Validator来校验
        result = execVal.validateReturnValue(invocation.getThis(), methodToValidate, returnValue, groups);
        //有异常直接抛出
        if (!result.isEmpty()) {
            throw new ConstraintViolationException(result);
        }
        return returnValue;
    }
}

基于XML的配置校验器

<mvc:annotation-driven validator="validator"/>

<!-- 配置验证器 -->
<bean id="validator" class="org.springframework.validation.beanvalidation.LocalValidatorFactoryBean">
  <property name="providerClass" value="org.hibernate.validator.HibernateValidator"/>
</bean>

<!--  如果要在控制器方法的参数上直接使用验证注解进行修饰,需要该配置 -->
<bean class="org.springframework.validation.beanvalidation.MethodValidationPostProcessor"/>

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1431060.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

如何在个人PC的桌面创建一个类似网吧的游戏菜单并分类?

GGTools 免费的桌面图标管理器、软件菜单、游戏菜单 单机版游戏菜单、个人/家用/家庭版游戏菜单、轻量级图标收纳软件

两台电脑怎么互传文件?4个方法提高传输效率!

“我在工作时总是需要将文件传给同小组的同事&#xff0c;想问问大家有什么两台电脑互传文件的简单操作方法吗&#xff1f;希望大家教教我呀&#xff01;” 在日常生活和工作中&#xff0c;我们经常需要将文件从一台电脑传输到另一台电脑。除了传统的U盘或移动硬盘等存储设备外…

【DBF格式转换器.exe】

一、概要 DBF文件是一种数据库文件格式&#xff0c;通常用于存储表格数据。这种文件格式曾经被广泛使用&#xff0c;尤其是在一些较旧的数据库系统中。然而&#xff0c;随着时间的推移&#xff0c;其他更现代的文件格式&#xff0c;如XLS&#xff08;Excel&#xff09;、CSV、D…

深入Spring MVC的工作流程

深入Spring MVC的工作流程 在Spring MVC的面试问题中&#xff0c;常常被询问到的一个问题。Spring MVC的程序中&#xff0c;HTTP请求是如何从开始到结束被处理的。为了研究这个问题&#xff0c;我们将需要深入学习一下Spring MVC框架的核心过程和工作流程。 1. 启动请求生命周…

代码随想录算法训练营29期|day39 任务以及具体任务

第九章 动态规划part02 62.不同路径 /*** 1. 确定dp数组下标含义 dp[i][j] 到每一个坐标可能的路径种类* 2. 递推公式 dp[i][j] dp[i-1][j] dp[i][j-1]* 3. 初始化 dp[i][0]1 dp[0][i]1 初始化横竖就可* 4. 遍历顺序 一行一行遍历* 5. 推导结果 。。。。。。。。** param m* p…

Windows内存管理 - 虚拟内存地址概念(Virtual Memory Address)

虽然可以寻址4GB的内存&#xff0c;而在PC里往往没有如此多的真实物理内存。操作系统和硬件&#xff08;这里指的是CPU中的内存管理单元MMU&#xff09;为使用者提供了虚拟内存的概念。Windows的所有程序&#xff08;包括Ring0层和Ring3层的程序&#xff09;可以操作的都是虚拟…

GPT3.5\GPT4系列计算完整prompt token数的官方方法

前言: ChatGPT如何计算token数&#xff1f;https://wtl4it.blog.csdn.net/article/details/135116493?spm1001.2014.3001.5502https://wtl4it.blog.csdn.net/article/details/135116493?spm1001.2014.3001.5502 GPT3.5\GPT4系列计算完整prompt token数的官方方法&#xff1…

航母编队反无人机蜂群作战能力需求分析

源自&#xff1a;指挥控制与仿真 作者&#xff1a;樊辉锦、巫银花、毕月、苏泽亚 “人工智能技术与咨询” 发布 声明:公众号转载的文章及图片出于非商业性的教育和科研目的供大家参考和探讨&#xff0c;并不意味着支持其观点或证实其内容的真实性。版权归原作者所有&#xff…

01 JDK的安装

JDK的安装 1 JDK的安装&#xff1a;参考&#xff1a; 1 JDK的安装&#xff1a; 说到Java&#xff0c;永远都有一个绕不开的话题&#xff0c;就是JDK(Java Development Kit)。JDK 是整个Java的核心&#xff0c;包括了Java运行环境&#xff0c;Java工具和Java基础的类库。安装JD…

来了来了,5000个红包封面免费领

今年公众号很慷慨&#xff0c;给长期运营的作者免费发放了6w个红包封面&#xff0c;感谢公众号平台。 往年还是自己花钱找别人设计&#xff0c;平台审核通过后才能正常发放给大家&#xff0c;自从AI绘画工具问世后&#xff0c;自己也能设计了。 下面的两个封面都是我用AI工具St…

05、全文检索 -- Solr -- Solr 全文检索之图形界面的文档管理(文档的添加、删除,如何通过关键字等参数查询文档)

目录 Solr 全文检索之文档管理添加文档使用 JSON 添加文档&#xff1a;使用 XML 添加文档: 删除文档使用 JSON 删除文档&#xff1a;使用 XML 删除文档&#xff1a; 查询文档查询文档的详细参数fq&#xff08;Filter Query&#xff09;&#xff1a;过滤sort&#xff1a;排序sta…

隧道穿透:常规反弹、加密反弹

目录 1、常规反弹 &#xff08;1&#xff09;Windows正向连接shell &#xff08;2&#xff09;Windows反向连接shell &#xff08;3&#xff09;Linux正向连接shell &#xff08;2&#xff09;利用Linux自带bash反弹Shell 2、加密反弹 1、常规反弹 假设在内网环境中发现…

寒假 day2

1、请编程实现单向循环链表的头插&#xff0c;头删、尾插、尾删 #include<stdio.h> #include<string.h> #include<stdlib.h> enum{FALSE-1,SUCCESS}; typedef int datatype; //定义节点结构体 //节点&#xff1a;数据域、指针域 typedef struct Node {//数…

机器学习系列——(九)决策树

简介 决策树作为机器学习的一种经典算法&#xff0c;在数据挖掘、分类和回归等任务中广泛应用。本文将详细介绍机器学习中的决策树算法&#xff0c;包括其原理、构建过程和应用场景。 原理 决策树是一种基于树状结构的监督学习算法&#xff0c;它通过构建一棵树来对数据进行分…

leetcode刷题(剑指offer)138.随机链表的复制

138.随机链表的复制 给你一个长度为 n 的链表&#xff0c;每个节点包含一个额外增加的随机指针 random &#xff0c;该指针可以指向链表中的任何节点或空节点。 构造这个链表的 深拷贝。 深拷贝应该正好由 n 个 全新 节点组成&#xff0c;其中每个新节点的值都设为其对应的原…

【Linux网络编程一】网络基础1(网络框架)

【Linux网络编程一】网络基础1&#xff08;网络框架&#xff09; 一.什么是协议1.通信问题2.协议本质3.网络协议标准 二.协议分层1.为什么协议要分层2.如何具体的分层 三.操作系统OS与网络协议栈的关系1.核心点&#xff1a;网络通信贯穿协议栈 四.局域网中通信的基本原理1.封装…

使用 IntelliJ IDEA 配合 Docker 对 Weblogic 中间件进行远程调试

使用idea对jar包远程调试&#xff1a; 打开一个springboot的项目进行远程调试设置&#xff1a; 运行&#xff1a; 其实我不太明白远程调试的意义&#xff0c;本地直接debug不好嘛。。。 点击debug的按钮&#xff0c;打断点测试&#xff1a; 跑到断点处&#xff1a; 远程de…

JDK17中的密封类sealed和permits使用指南:什么是Java中的sealed和permits?

博主猫头虎的技术世界 &#x1f31f; 欢迎来到猫头虎的博客 — 探索技术的无限可能&#xff01; 专栏链接&#xff1a; &#x1f517; 精选专栏&#xff1a; 《面试题大全》 — 面试准备的宝典&#xff01;《IDEA开发秘籍》 — 提升你的IDEA技能&#xff01;《100天精通鸿蒙》 …

MYSQL——MySQL8.3无法启动

在新电脑上装了个MySQL&#xff0c;但是无法使用net start mysql启动&#xff0c;很是纳闷&#xff0c;使用mysqld --console去查看报错&#xff0c;也是没报错的&#xff0c;但是奇怪的是&#xff0c;我输入完这个mysqld --console之后&#xff0c;就等于启动了mysql了&#x…

一步步成为React全栈大师:从环境搭建到应用部署

文章目录 第一步&#xff1a;环境搭建第二步&#xff1a;了解React基础第三步&#xff1a;组件与路由第四步&#xff1a;状态管理第五步&#xff1a;接口与数据交互第六步&#xff1a;样式与布局第七步&#xff1a;测试第八步&#xff1a;构建与部署《深入浅出React开发指南》内…