SpringBoot的validation参数校验

文章目录
  • 前言
  • 一、引入validation 依赖
  • 二、validation中的注解说明
    • (1)@Validated
    • (2)@Valid
    • (3)@NotNull
    • (4)@NotBlank
    • (5)@NotEmpty
    • (6)@Pattern
    • (7) @Email
    • (8)@Size
  • 三、validation的使用
    • 1.一般校验
    • 2.嵌套验证
    • 3.分组校验
    • 4.自定义校验注解
    • 5.反射加自定义注解实现动态校验
  • 四、validation 校验失败处理
  • 总结

前言

在项目开发过程,后端经常会对前端传递的参数进行各种校验,只有在校验通过时,才会执行后续的业务代码,否则抛出异常信息给前端。当参数较多时,刚开始会使用大量的 if…else…来逐一对参数进行检验,或则使用策略模式来方法来减少 if…else…的检验代码。但是将这些代码写在控制层或则业务层都会让代码过于臃肿。而且如果后期参数校验变了,或则参数不需要校验时,还需要大量删除,或修改检验参数的逻辑代码,因此采用 validation来对参数进行校验简化代码。


一、引入validation 依赖

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

二、validation中的注解说明

(1)@Validated

用于在 Spring 控制器层方法参数进行验证

@PostMapping("/validator")
public String ValidatorTest(@RequestBody @Validated UserParam user) {return "success";}

(2)@Valid

是 Java 标准的一部分,用于声明需要对嵌套对象进行递归验证。在 Spring 框架中,通常用于处理复杂对象结构的验证,如验证对象中的集合或对象引用的属性

@Data
public class EmpParam  implements Serializable {@NotBlank(message = "员工姓名不能为空")private String empName;@NotBlank(message = "员工手机号不能为空")private String empPhone;@Valid@NotNull(message = "部门信息不能为空")private DeptParam deptParam;
}@Data
public class DeptParam implements Serializable {@NotNull(message = "部门ID不能为空")private Long deptId;@NotBlank(message = "部门名称不能为空")private String deptName;
}

(3)@NotNull

注解用于检查被注解的元素值不为 null。适用于字符串、集合、Map 等任何对象类型,但不适用于基本数据类型(如 int、long 等)
常用于检查对象是否为 null

@NotNull(message = "员工ID不能为空")
private Long empId;@NotNull(message = "部门信息不能为空")
private DeptParam deptParam;

(4)@NotBlank

用于检查被注解的字符串元素不为 null 且去除两端空白字符后长度大于 0。只适用于字符串类型。

@NotBlank(message = "员工身份证号不能为空")
private String empIdCard;

(5)@NotEmpty

注解用于检查被注解的元素不为 null 且不为空,适用于字符串、集合、Map 等。
如果是字符串,相当于同时检查 null 和长度大于 0

@NotEmpty
private List<String> emails;

(6)@Pattern

指定字段必须符合指定的正则表达式

@Pattern(regexp ="^1[3|4|5|6|7|8|9][0-9]d{8}$")
private String phone

(7) @Email

指定字段必须符合Email格式。

@Email
private String email;

(8)@Size

指定字段的长度范围

@Size(min = 6, max = 20)
private String password;

三、validation的使用

1.一般校验

一般校验:这里指的是,后端参数是一个实体类对象来接收参数,并且校验实体类中各个属性(被注解标注过的)

@RestController
@RequestMapping("/test")
public class ValidatorTestController {@PostMapping("/validator")public String ValidatorTest(@RequestBody @Validated UserParam user) {UserParam contractParam = new UserParam();BeanUtils.copyProperties(user, contractParam);System.out.println(contractParam);return "success";}
}

实体类中的属性分别添加注解进行校验

@Data
public class UserParam {@NotBlank(message = "用户名不能为空")private String username;@NotBlank(message = "密码不能为空")@Size(min = 6, max = 20, message = "密码长度在6-20位之间")private String password;@NotBlank(message = "手机号不能为空")private String phone;@NotBlank(message = "身份证不能为空")private String idCard;@NotBlank(message = "身份证姓名不能为空")private String idCardName;@NotBlank(message = "车牌号不能为空")private String carNumber;@NotBlank(message = "银行卡号不能为空")private String bankCardNumber;}

用postman测试注解是生效的。注意响应的结果,是将所有参数没校验通过都一并响应给前端,方便前端处理。
在这里插入图片描述

2.嵌套验证

嵌套检验:后端接收参数是个实体类对象,并且实体类对象中还有一个实体或者是一个List类型对象
比如:添加员工信息时,为员工分配部门
控制层一定得使用 @Validated标注参数

@RestController
@RequestMapping("/emp")
public class EmpController {@PostMapping("/add")public String addEmp(@RequestBody @Validated EmpParam empParam) {System.out.println(empParam);return "success";}
}

员工实体类参数:DeptParam 必须加上 @Valid注解才能校验 DeptParam 中的字段(被注解标注的)

@Data
public class EmpParam  implements Serializable {@NotBlank(message = "员工姓名不能为空")private String empName;@NotBlank(message = "员工手机号不能为空")private String empPhone;@NotBlank(message = "员工邮箱不能为空")private String empEmail;@NotBlank(message = "员工身份证号不能为空")private String empIdCard;@Valid@NotNull(message = "部门信息不能为空")private DeptParam deptParam;
}@Data
public class DeptParam implements Serializable {@NotNull(message = "部门ID不能为空")private Long deptId;@NotBlank(message = "部门名称不能为空")private String deptName;
}

使用postman测试
在这里插入图片描述

3.分组校验

分组校验指的是,不通场景,参数检验方式不同。比如添加员工时候,由于empId在数据库是自增的,所以添加员工时 empParam 参数中empId 可以为空,但是在修改用户时,就要求empId不能为空。前提是 emp 新增和修改接口来接收前端的参数都是同一个 empParam 类

(1)自定义分组接口

public interface ValidationGroup {interface Create extends Default{}interface Update extends Default{}
}

(2)实体类中属性添加校验分组

@Data
public class EmpParam  implements Serializable {@NotNull(message = "员工ID不能为空", groups = {ValidationGroup.Update.class}) //注意查看添加接口和修改接口验证分组校验是否正确private Long empId;@NotBlank(message = "员工姓名不能为空",groups = {ValidationGroup.Create.class})private String empName;@NotBlank(message = "员工手机号不能为空")private String empPhone;@NotBlank(message = "员工邮箱不能为空")private String empEmail;@NotBlank(message = "员工身份证号不能为空")private String empIdCard;@Valid@NotNull(message = "部门信息不能为空")private DeptParam deptParam;
}

(3)控制层测试类

@RestController
@RequestMapping("/emp")
public class EmpController {@PostMapping("/add")public String addEmp(@RequestBody @Validated({ValidationGroup.Create.class}) EmpParam empParam) {System.out.println(empParam);return "success";}@PutMapping("/update")public String updateEmp(@RequestBody @Validated(ValidationGroup.Update.class) EmpParam empParam) {System.out.println(empParam);return "success";}
}

(4)使用posman测试
在这里插入图片描述

在这里插入图片描述

4.自定义校验注解

从上述方法中发现 validation这个框架所提供的校验注解,只是一些基本的判空,字符长度的校验。在日常开发中可能需要校验:比如手机号不能为空同时,必须是11位数字,以及添加用户时,用户手机号必须唯一(需要查询数据库)等等。显然是不满足日常开发的需求。因此我们可以根据 自身的需求来做校验注解。

接下来我们将自定义一个注解来专门校验身份证号

@Constraint(validatedBy = IdCardNumberValidator.class)  // @IdCardNumber 注解校验类的具体实现类
@Target({ElementType.FIELD,ElementType.METHOD}) // 表示注解可以应用于字段和方法上
@Retention(RetentionPolicy.RUNTIME) // 表示注解在运行时保留,因此可以通过反射机制读取。强调,可以通过反射获取,
public @interface IdCardNumber {String message() default "身份证错误";Class<?>[] groups() default {};Class<?>[] payload() default {};
}

IdCardNameValidator 校验类来实现 ConstraintValidator接口即可

import com.personal.validation.annotations.IdCardName;
import com.personal.validation.utils.ValidatorUtils;
import jakarta.validation.ConstraintValidator;
import jakarta.validation.ConstraintValidatorContext;public class IdCardNameValidator implements ConstraintValidator<IdCardName, String> {@Overridepublic void initialize(IdCardName constraintAnnotation) {ConstraintValidator.super.initialize(constraintAnnotation);}@Overridepublic boolean isValid(String idCardName, ConstraintValidatorContext constraintValidatorContext) {return ValidatorUtils.isValidName(idCardName);}
}

校验工具类

public class ValidatorUtils {//判断身份证号是否有效/*** '^':表示匹配字符串的开头* '[1-9]':表示匹配1-9之间的数字,确保身份证号码的前6位不为0* '\d{5}':表示匹配5位数字,确保身份证号码的前6位为数字,用于匹配地区码* '(18|19|([23]\d))':表示匹配18或19或20-23之间的数字,确保身份证号码的前2位为18或19或20-23之间的数字,用于匹配年份* '\d{2}':表示接下来的2位是月份,范围是01-12之间的数字*  '((0[1-9])|(1[0-2]))':表示匹配月份,范围是01-12之间的数字*  '(([0-2][1-9])|10|20|30|31)':表示日期,前两位0-2表示01-29,10、20、30、31分别单独列出*  '\d{3}':表示匹配3位数字,确保身份证号码的第18位为数字,用于匹配顺序码* '[0-9Xx]':表示匹配0-9或X或x,确保身份证号码的第17位为数字或X或x,用于匹配校验位** @param idCard* @return*/public static boolean isValidIdCard(String idCard) {return Pattern.matches("^[1-9]\d{5}(18|19|([23]\d))\d{2}((0[1-9])|(10|11|12))(([0-2][1-9])|10|20|30|31)\d{3}[0-9Xx]$", idCard);}
}

实体类上加上 @IdCardNumber 注解

@Data
public class UserParam {@NotBlank(message = "用户名不能为空")private String username;@NotBlank(message = "密码不能为空")@Size(min = 6, max = 20, message = "密码长度在6-20位之间")private String password;@NotBlank(message = "手机号不能为空")private String phone;@NotBlank(message = "身份证不能为空")@IdCardNumber(message = "身份证不正确")// 自定义注解来校验,并不会跟原有注解冲突private String idCard;@NotBlank(message = "身份证姓名不能为空")private String idCardName;@NotBlank(message = "车牌号不能为空")private String carNumber;@NotBlank(message = "银行卡号不能为空")private String bankCardNumber;}

测试自定义注解是生效的,虽然身份证没空,但是校验没能通过的自定义注解的校验类

在这里插入图片描述
以上只是简单写了一个案例,如何自定义一个注解,以后可以自定义一个注解来校验前端传递过来的参数必须在数据库是唯一的,例如手机号,不能多个用户使用同个手机号,身份证也是如此。

5.反射加自定义注解实现动态校验

现在有这样一个场景,系统中客户表中有两个类型客户:个体工商户和企业客户两类,系统的客户认证只是一个接口,但是呢,企业客户和个体客户校验信息不一样,企业校验是,企业名称,企业信用代码,法人,营业执照等。个体工商户验证时客户身份证,名字,电话等。简单来说就是通过一个 authType 认证类型 字段 来判断那些那些参数不能为空

(1)添加客户实体类参数
如果前端传递的 authType =1,就校验 SingleCustomerParam 对象不能为空,并且 SingleCustomerParam 类中被 @NotBlank注解标注也不能为空。如果authType = 2,就检验 FirmCustomerParam 对象不能为空,并且 FirmCustomerParam 类中被 @NotBlank注解标注也不能为空。如果authType != 1 或则 2 就返回认证类型错误给前端

@Data
public class AddCustomerParam implements Serializable {/***  客户类型 1个体工商户 2:企业客户*/@NotBlank(message = "客户类型不能为空")private String authType;/*** 1个体工商户*/private SingleCustomerParam singleCustomerParam;/*** 2企业客户*/private FirmCustomerParam firmCustomerParam;
}

SingleCustomerParam 实体类参数类

@Data
public class SingleCustomerParam implements Serializable {/*** 客户名称*/
//    @NotBlank(message = "客户名称不能为空") //private String customerName;/*** 客户身份证照片*/@NotBlank(message = "客户身份证照片不能为空")private String idCardImg;/*** 认证图片和视频*/@NotBlank(message = "认证图片和视频不能为空")private String verifyImg;/*** 客户真是姓名*/@NotBlank(message = "客户真是姓名不能为空")private String realName;/*** 客户身份证号*/@NotBlank(message = "客户身份证号不能为空")private String idCard;/*** 客户手机号*/@NotBlank(message = "客户手机号不能为空")private String phone;
}

FirmCustomerParam 实体参数类

@Data
public class FirmCustomerParam implements Serializable {/*** 企业营业执照*/@NotBlank(message = "企业营业执照不能为空")private String businessLicense;/*** 企业名称*/@NotBlank(message = "企业名称不能为空")private String firmName;/*** 企业信用代码*/@NotBlank(message = "企业信用代码不能为空")private String creditCode;/*** 企业地址*/@NotBlank(message = "企业地址不能为空")private String address;/*** 企业联系人*/@NotBlank(message = "企业联系人不能为空")private String linkman;/*** 企业联系人电话*/@NotBlank(message = "企业联系人电话不能为空")private String mobile;}

(2)自定义参数校验注解 @CustomerParamTwo

@Target({ElementType.TYPE})
@Retention(RetentionPolicy.RUNTIME)
@Constraint(validatedBy = CustomerParamValidatorTwo.class)//注解校验类
public @interface CustomerParamTwo {String message() default "Invalid customer parameters";Class<?>[] groups() default {};Class<? extends Payload>[] payload() default {};
}

CustomerParamValidatorTwo 实现 ConstraintValidator接口 重写具体校验规则

import com.personal.validation.annotations.CustomerParamTwo;
import com.personal.validation.param.AddCustomerParam;
import com.personal.validation.param.FirmCustomerParam;
import com.personal.validation.param.SingleCustomerParam;
import com.personal.validation.utils.ValidatorUtils;
import jakarta.validation.ConstraintValidator;
import jakarta.validation.ConstraintValidatorContext;
import jakarta.validation.constraints.NotBlank;import java.lang.reflect.Field;public class CustomerParamValidatorTwo implements ConstraintValidator<CustomerParamTwo, AddCustomerParam> {@Overridepublic void initialize(CustomerParamTwo constraintAnnotation) {ConstraintValidator.super.initialize(constraintAnnotation);}@Overridepublic boolean isValid(AddCustomerParam customerParam, ConstraintValidatorContext constraintValidatorContext) {String authType =customerParam.getAuthType();if ("1".equals(authType)) {if(customerParam.getSingleCustomerParam() instanceof SingleCustomerParam){return customerParam.getSingleCustomerParam() != null&& isValidSingleCustomerParam(customerParam.getSingleCustomerParam(), constraintValidatorContext);}else//当authType = 1 时,singleCustomerParam不能为空addConstraintViolation(constraintValidatorContext, "singleCustomerParam","个人客户信息不能为空");return false;}else if ("2".equals(authType)) {if(customerParam.getFirmCustomerParam() instanceof FirmCustomerParam){return customerParam.getFirmCustomerParam() != null&& isValidFirmCustomerParam(customerParam.getFirmCustomerParam(), constraintValidatorContext);}else//当authType = 2 时,firmCustomerParam不能为空addConstraintViolation(constraintValidatorContext, "firmCustomerParam","企业客户信息不能为空");return false;}//单独校验authType类型if(!("1".equals(authType) || "2".equals(authType) || "".equals(authType))){addConstraintViolation(constraintValidatorContext, "authType","认证类型不正确");return false;}return false;}private boolean isValidSingleCustomerParam(SingleCustomerParam param, ConstraintValidatorContext context) {boolean isValid = true ;Field[] fields = SingleCustomerParam.class.getDeclaredFields();//反射获取对象的所有字段for (Field field : fields) {if (field.isAnnotationPresent(NotBlank.class)) {//获取属性上是否标注了校验注解// 如果字段有 @NotBlank 注解,则进行非空验证if (!validateField(param, field, context)) {isValid = false;}}}if(!ValidatorUtils.isValidPhoneNumber(param.getPhone())){addConstraintViolation(context, "phone","手机号不正确");isValid = false;}return isValid;}private boolean isValidFirmCustomerParam(FirmCustomerParam param, ConstraintValidatorContext context) {boolean isValid = true ;Field[] fields = FirmCustomerParam.class.getDeclaredFields();for (Field field : fields) {if (field.isAnnotationPresent(NotBlank.class)) {// 如果字段有 @NotBlank 注解,则进行非空验证if (!validateField(param, field, context)) {isValid = false;}}}return isValid;}// 字段校验不通过后,获取校验错误信息private boolean validateField(Object param, Field field, ConstraintValidatorContext context) {try {field.setAccessible(true);Object value = field.get(param);if (value == null || value.toString().trim().isEmpty()) {String message = field.getAnnotation(NotBlank.class).message();addConstraintViolation(context, field.getName(),message);return false;}return true;} catch (IllegalAccessException e) {return false;}}//将所有校验不通过的错误信息,封装返回给前端private void addConstraintViolation(ConstraintValidatorContext context, String field, String message) {context.disableDefaultConstraintViolation();context.buildConstraintViolationWithTemplate(message).addPropertyNode(field).addConstraintViolation();}
}

校验控制层接口类

@RestController
@RequestMapping("/customer")
public class CustomerController {/*** @param addCustomerParam* @return*/@PostMapping("/add")public String addCustomer(@RequestBody @Validated AddCustomerParam addCustomerParam) {System.out.println(addCustomerParam);// 客户认证参数全部交给自定义注解校验,我们后续只关注业务代码,将 检验逻辑 与 业务代码逻辑 分离return "add customer success";}
}

在校验类加上自定义的注解 @CustomerParamTwo

@Data
@CustomerParamTwo
public class AddCustomerParam implements Serializable {/***  客户类型 1个体工商户 2:企业客户*/@NotBlank(message = "客户类型不能为空")private String authType;/*** 1个体工商户*/private SingleCustomerParam singleCustomerParam;/*** 2企业客户*/private FirmCustomerParam firmCustomerParam;
}

接下来我们进行测试:
测试authType 为空 或则 authType !=1 或则 2
在这里插入图片描述
在这里插入图片描述

authType = 1 校验 singleCustomerParam 个体工商户参数
在这里插入图片描述
校验成功:
在这里插入图片描述

authType = 2 校验 firmCustomerParam 企业客户参数
在这里插入图片描述
在这里插入图片描述

在这里插入图片描述

四、validation 校验失败处理

从上述案例中,校验失败错误信息原本抛出的异常不是如此的,validation 校验异常需要自己手动拦截并封装错误信息返回。接下来将如何捕获 validation 的异常信息,并封装成一个对象返回给前端
(1)自定义封装结果类

public class R {private int code;private String message;private Object data;public R() {}public R(int code, String message) {this.code = code;this.message = message;}public static R ok() {return new R(200, "Success");}public static R fail(int code, String message) {return new R(code, message);}// Getters and setters// You may want to add additional methods for data handlingpublic int getCode() {return code;}public void setCode(int code) {this.code = code;}public String getMessage() {return message;}public void setMessage(String message) {this.message = message;}public Object getData() {return data;}public void setData(Object data) {this.data = data;}
}

(2)自定义ValidationException 异常

public class ValidationException extends RuntimeException {public ValidationException(String message) {super(message);}
}

(3)自定义拦截器
直接在官网复制 validation 拦截器

@ControllerAdvice
public class CustomGlobalExceptionHandler {// Handle ConstraintViolationException@ExceptionHandler(ConstraintViolationException.class)@ResponseStatus(HttpStatus.BAD_REQUEST)@ResponseBodypublic R handleValidationException(ConstraintViolationException ex) {List<String> errors = new ArrayList<>();for (ConstraintViolation<?> violation : ex.getConstraintViolations()) {errors.add(violation.getPropertyPath() + ": " + violation.getMessage());}return R.fail(HttpStatus.BAD_REQUEST.value(), errors.toString());}// Handle MethodArgumentTypeMismatchException@ExceptionHandler(MethodArgumentTypeMismatchException.class)@ResponseStatus(HttpStatus.BAD_REQUEST)@ResponseBodypublic R handleMethodArgumentTypeMismatchException(MethodArgumentTypeMismatchException ex) {String error = ex.getName() + " should be of type " + ex.getRequiredType().getName();return R.fail(HttpStatus.BAD_REQUEST.value(), error);}// Handle BindException@ExceptionHandler(BindException.class)@ResponseStatus(HttpStatus.BAD_REQUEST)@ResponseBodypublic R handleBindException(BindException ex) {List<String> errors = new ArrayList<>();for (FieldError error : ex.getFieldErrors()) {errors.add(error.getField() + ": " + error.getDefaultMessage());}return R.fail(HttpStatus.BAD_REQUEST.value(), errors.toString());}// Handle custom ValidationException@ExceptionHandler(ValidationException.class)  // 在这里进行拦截的@ResponseStatus(HttpStatus.BAD_REQUEST)@ResponseBodypublic R handleValidationException(ValidationException ex) {return R.fail(HttpStatus.BAD_REQUEST.value(), ex.getMessage()); //拦截运行错误时的状态吗以及错误信息}// Handle generic exceptions@ExceptionHandler(Exception.class)@ResponseStatus(HttpStatus.INTERNAL_SERVER_ERROR)@ResponseBodypublic R handleException(Exception ex) {return R.fail(HttpStatus.INTERNAL_SERVER_ERROR.value(), ex.getMessage());}
}

总结

例如:以上就是关于日常开发过程参数校验及封装错误信息返回给前端

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

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

相关文章

SSH克隆github项目

1、生成密钥 ssh-keygen -t rsa -C "你的邮箱xxx.com" 全程回车即可&#xff08;不用输入ras文件名及密码&#xff09;、为了方便下面的公钥查看 2、配置公钥 查看公钥内容 cat c:\Users\xxx\.ssh\id_rsa.pub(修改为自己的路径及名字) 将公钥内容复制并粘贴至…

CASAIM与中国航天携手合作,CASAIM IS全自动化光学测量系统交付中国航天山西工厂,助力航空航天零部件全自动化3D测量

近日&#xff0c;CASAIM与中国航天达成全自动化光学测量技术合作&#xff0c;并将CASAIM IS全自动化光学测量系统交付给中国航天科技集团山西工厂&#xff0c;这一合作标志着双方在智能制造和精密测量领域迈出了重要一步。 中国航天科技&#xff0c;是在中国战略高技术领域拥有…

基于Springboot+Vue的电子博物馆系统

基于SpringbootVue的电子博物馆系统 前言&#xff1a;随着信息技术的不断发展&#xff0c;传统博物馆的参观方式逐渐向数字化、在线化转型。电子博物馆作为这一转型的重要组成部分&#xff0c;能够通过信息化手段为用户提供更丰富、更便捷的博物馆参观体验。本文基于Spring Boo…

在Docker中部署禅道,亲测可用

1、确保centos中已安装docker docker -v 2、启动docker systemctl start docker 3、可设置docker开机启动 systemctl enable docker.service 4、获取最新版禅道开源版镜像 docker pull idoop/zentao 5、运行镜像生成禅道容器【创建 /data/www /data/data 目录】 doc…

vitepress组件库文档项目 markdown语法大全(修正版)

#上次总结的 有些语法是用在markdown文档中的 使用到vitepress项目中有些语法可能有出入 于是我再总结一版 vitepress项目中的markdown语法大全 在阅读本章节之前&#xff0c;请确保你已经对 Markdown 有所了解。如果你还不了解 Markdown &#xff0c;请先学习一些Markdown 教…

Blender导入下载好的fbx模型像的骨骼像针戳/像刺猬

为什么我下载下来的骨骼模型和我自己绑定的模型骨骼朝向完全不一样 左边是下载的模型 右边是我自己绑定的模型 左边的模型刚刚感觉都是像针一样往外戳的&#xff0c;像刺猬一样那种。 解决方法勾选自动骨骼坐标系

ASP.NET CORE API 解决跨域问题

环境 vs2022 .net 8 创建ASP.net Core API项目 配置跨域 编写ApiController 启动项目 得到服务器运行的 地址 在Hbuiler中创建web项目&#xff0c;编写代码 【运行】-【运行到浏览器】-选择一个浏览器,查看结果 正常显示 问题 如果允许所有源访问&#xff0c;有安全风险方…

【AI系统】MobileFormer

MobileFormer 在本文中&#xff0c;将介绍一种新的网络-MobileFormer&#xff0c;它实现了 Transformer 全局特征与 CNN 局部特征的融合&#xff0c;在较低的成本内&#xff0c;创造一个高效的网络。通过本节&#xff0c;让大家去了解如何将 CNN 与 Transformer 更好的结合起来…

决策树:ID3、C4.5和CART特征选择方式

1 前言 该文章主要目的是记录ID3、C4.5和CART特征选择方式&#xff0c;这里只对决策树进行简单介绍。 决策树&#xff08;Decision Tree&#xff09;算法是一种有监督学习算法&#xff0c;它利用分类的思想&#xff0c;根据数据的特征构建数学模型&#xff0c;从而达到数据的筛…

【3D AIGC】Img-to-3D、Text-to-3D、稀疏重建(2024年文章汇总)

文章目录 1. Wonderworld&#xff1a;拓展图片边界&#xff0c;生成3D场景2. 3DTopia-XL&#xff1a;扩散模型辅助生成3. 3DGS-Enhancer: 通过视图一致2D Diffusion&#xff0c;提升无界3D Gaussian Splatting (NlPs2024 Spotlight)4. L3DG&#xff1a;Latent 3D Gaussian Diff…

三款电容麦的对比

纸面参数 第一款麦克风 灵敏度: -36 dB 2 dB&#xff08;0 dB1V/Pa at 1 kHz&#xff09; 灵敏度较低&#xff0c;需要更高的增益来拾取同样的音量。频率响应: 40 Hz - 18 kHz 响应范围较窄&#xff0c;尤其在高频区域。等效噪音级: ≤18 dB&#xff08;A计权&#xff09; 噪…

运行 GreatSQL 时为什么要求关闭透明大页

在大部分运维规范中&#xff0c;一般都会要求在运行 GreatSQL/MySQL 的环境中要关闭透明大页&#xff0c;那么到底什么是透明大页&#xff0c;为什么要关闭&#xff0c;打开有什么风险吗&#xff1f; 在此之前&#xff0c;我也是有点懵的&#xff0c;本文试着回答这个疑问&…

JUnit介绍:单元测试

1、什么是单元测试 单元测试是针对最小的功能单元编写测试代码&#xff08;Java 程序最小的功能单元是方法&#xff09;单元测试就是针对单个Java方法的测试。 2、为什么要使用单元测试 确保单个方法运行正常&#xff1b; 如果修改了代码&#xff0c;只需要确保其对应的单元…

Elasticsearch:使用硬件加速的 SIMD 指令实现超快 BBQ

作者&#xff1a;来自 Elastic Chris Hegarty 我们如何使用硬件加速 SIMD&#xff08;Single Instruction Multiple Data - 单指令多数据&#xff09;指令优化 BBQ 中的向量比较。 随着我们继续致力于让 Elasticsearch 和 Apache Lucene 成为存储和搜索向量数据的最佳场所&…

青龙面板添加任务执行自己的脚本文件(非订阅) 保姆级图文

目录 效果预览脚本存放的位置创建任务cron规则字段含义&#xff1a;常见的特殊字符&#xff1a; 可能你的脚本需要安装依赖总结 欢迎关注 『青龙面板』 专栏&#xff0c;持续更新中 欢迎关注 『青龙面板』 专栏&#xff0c;持续更新中 效果预览 你的python脚本 print(123)运行…

flink的安装配置(详细版本)

Standalone集群模式安装部署 conda deactivate 退出 base环境 Flink支持多种安装模式。 local&#xff08;本地&#xff09;——本地模式 standalone——独立模式&#xff0c;Flink自带集群&#xff0c;开发测试环境使用 standaloneHA—独立集群高可用模式&#xff0c;Fli…

Linux Cgroup学习笔记

文章目录 Cgroup(Control Group)引言简介Cgroup v1通用接口文件blkio子系统cpu子系统cpuacct子系统cpuset子系统devices子系统freezer子系统hugetlb子系统memory子系统net_cls子系统net_prio子系统perf_event子系统pids子系统misc子系统 Cgroup V2基础操作组织进程和线程popula…

JVM, JRE 和 JDK

JRE: Java Runtime Environment, Java 运行环境. JDK: Java Development Kit, Java 开发工具包. JRE JVM 核心类库 运行工具 JDK JVM 核心类库 开发工具 JVM: Java Virtual Machine, Java 虚拟机. 核心类库: Java 已经写好的东西, 直接拿来用即可. 开发工具: 包括 …

用于LiDAR测量的1.58um单芯片MOPA(一)

--翻译自M. Faugeron、M. Krakowski1等人2014年的文章 1.简介 如今&#xff0c;人们对高功率半导体器件的兴趣日益浓厚&#xff0c;这些器件主要用于遥测、激光雷达系统或自由空间通信等应用。与固态激光器相比&#xff0c;半导体器件更紧凑且功耗更低&#xff0c;这在低功率供…

前端框架的选择与反思:在简约与复杂之间寻找平衡

在当今互联网时代&#xff0c;前端开发已经成为web应用构建中不可或缺的一环。从最初的静态HTML页面&#xff0c;到如今复杂的单页应用&#xff08;SPA&#xff09;&#xff0c;前端技术的发展让我们见证了Web应用的蓬勃发展。然而&#xff0c;伴随着技术的进步&#xff0c;一个…