Spring中的数据校验
数据校验基础
参考: Java Bean Validation 规范
Spring对Bean Validation的支持
Spring定义了一个接口org.springframework.validation.Validator,用于应用相关的对象的校验器。
这个接口完全从基础设施或者上下文中脱离的,这意味着它没有跟web层或者数据访问层或者其余任何的某一个层次发生耦合。所以它能用于应用中的任意一个层次,能对应用中的任意一个对象进行校验。
接口定义
public interface Validator {// 此clazz是否可以被validateboolean supports(Class<?> clazz);// 执行校验,错误消息放在Errors中// 如果能执行校验,通常也意味着supports方法返回true// 可以参考ValidationUtils这个工具类void validate(Object target, Errors errors);
}
接口实现

SmartValidator
对Validator接口进行了增强,能进行分组校验。
public interface SmartValidator extends Validator {// validationHints:就是启动的校验组// target:需要校验的结果// errors:封装校验void validate(Object target, Errors errors, Object... validationHints);// 假设value将被绑定到指定对象中的指定字段上,并进行校验// @since 5.1 这个方法子类需要复写 否则不能使用default void validateValue(Class<?> targetType, String fieldName, @Nullable Object value, Errors errors, Object... validationHints) {throw new IllegalArgumentException("Cannot validate individual value for " + targetType);}
}
SpringValidatorAdapter
SpringValidatorAdapter实现了对 Bean Validation的适配。
// 可以看到,这个接口同时实现了Spring中的SmartValidator接口跟JSR中的Validator接口
public class SpringValidatorAdapter implements SmartValidator, javax.validation.Validator {//约束必须使用的3个属性private static final Set<String> internalAnnotationAttributes = new HashSet<>(4);static {internalAnnotationAttributes.add("message");internalAnnotationAttributes.add("groups");internalAnnotationAttributes.add("payload");}// targetValidator就是实际完成校验的对象@Nullableprivate javax.validation.Validator targetValidator;public SpringValidatorAdapter(javax.validation.Validator targetValidator) {Assert.notNull(targetValidator, "Target Validator must not be null");this.targetValidator = targetValidator;}SpringValidatorAdapter() {}void setTargetValidator(javax.validation.Validator targetValidator) {this.targetValidator = targetValidator;}// 支持对所有类型的Bean的校验@Overridepublic boolean supports(Class<?> clazz) {return (this.targetValidator != null);}// 调用targetValidator完成校验,并通过processConstraintViolations方法封装校验后的结果到Errors中@Overridepublic void validate(Object target, Errors errors) {if (this.targetValidator != null) {processConstraintViolations(this.targetValidator.validate(target), errors);}}// 完成分组校验@Overridepublic void validate(Object target, Errors errors, Object... validationHints) {if (this.targetValidator != null) {processConstraintViolations(this.targetValidator.validate(target, asValidationGroups(validationHints)), errors);}}// 完成对对象上某一个字段及给定值的校验@SuppressWarnings("unchecked")@Overridepublic void validateValue(Class<?> targetType, String fieldName, @Nullable Object value, Errors errors, Object... validationHints) {if (this.targetValidator != null) {processConstraintViolations(this.targetValidator.validateValue((Class) targetType, fieldName, value, asValidationGroups(validationHints)), errors);}}// @since 5.1// 将validationHints转换成JSR中的分组private Class<?>[] asValidationGroups(Object... validationHints) {Set<Class<?>> groups = new LinkedHashSet<>(4);for (Object hint : validationHints) {if (hint instanceof Class) {groups.add((Class<?>) hint);}}return ClassUtils.toClassArray(groups);}// 省略对校验错误的封装// .....// 省略对JSR中validator接口的实现,都是委托给targetValidator完成的// ......}
ValidatorAdapter
跟SpringValidatorAdapter同一级别的类,但是不同的是他没有实现JSR中的Validator接口。一般不会使用这个类。
它实现了对 SmartValidator 的适配。
public class ValidatorAdapter implements SmartValidator, ApplicationContextAware, InitializingBean, DisposableBean {private final SmartValidator target;private final boolean existingBean;ValidatorAdapter(SmartValidator target, boolean existingBean) {this.target = target;this.existingBean = existingBean;}public final Validator getTarget() {return this.target;}@Overridepublic boolean supports(Class<?> clazz) {return this.target.supports(clazz);}//... ...
}
CustomValidatorBean
配置一个bean,暴露一个 JSR-303 Validator,使用了 JSR 的3个接口。
public class CustomValidatorBean extends SpringValidatorAdapter implements Validator, InitializingBean {@Nullableprivate ValidatorFactory validatorFactory;@Nullableprivate MessageInterpolator messageInterpolator;@Nullableprivate TraversableResolver traversableResolver;
}
LocalValidatorFactoryBean
OptionalValidatorFactoryBean
继承了LocalValidatorFactoryBean,区别在于让校验器的初始化成为可选的,即使校验器没有初始化成功也不会报错。
@Validated跟@Valid的区别
定义
@Target({ElementType.TYPE, ElementType.METHOD, ElementType.PARAMETER})
@Retention(RetentionPolicy.RUNTIME)
@Documented
public @interface Validated {// 校验时启动的分组Class<?>[] value() default {};
}
@Target({ METHOD, FIELD, CONSTRUCTOR, PARAMETER, TYPE_USE })
@Retention(RUNTIME)
@Documented
public @interface Valid {//没有提供任何属性
}
区别
-
来源不同:
@Valid是JSR的规范,来源于javax.validation包下,而@Validated是Spring自身定义的注解,位于org.springframework.validation.annotation包下 -
作用范围不同:
@Validated无法作用在字段上,正因为如此它就无法完成对级联属性的校验。而@Valid的没有这个限制。 -
注解中的属性不同:
@Validated注解中可以提供一个属性去指定校验时采用的分组,而@Valid没有这个功能,因为@Valid不能进行分组校验
应用
准备
待校验的类:
@Data
public class ValidatedData {@NotNullpublic String name;@Positivepublic Integer age;@NotNull@NotEmptyprivate List<@Email String> emails;/*** 定义的2个组,以接口的形式*/public interface GroupA {}public interface GroupB {}}/*** 外部验证数据,用于级联验证** @author lihz* @date 2023/2/18*/
@Data
public class OuterValidatedData {@NotNullString name;@ValidValidatedData validatedData;
}
测试Controller:
package com.jurassic.cloud.project.controller;import com.jurassic.cloud.project.dto.OuterValidatedData;
import com.jurassic.cloud.project.dto.ValidatedData;
import com.jurassic.cloud.project.service.impl.ValidationService;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.validation.annotation.Validated;
import org.springframework.web.bind.annotation.PostMapping;
import org.springframework.web.bind.annotation.RequestBody;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;import javax.validation.Valid;
import javax.validation.constraints.Max;
import javax.validation.constraints.NotBlank;@RestController
@RequestMapping("/validate")
public class ValidationController {@AutowiredValidationService validationService;@PostMapping("/valid")public String testValid(@RequestBody @Valid ValidatedData data) {System.out.println(data);return "OK";}@PostMapping("/validated")public String testValidated(@RequestBody @Validated ValidatedData data) {System.out.println(data);return "OK";}@PostMapping("/valid/nest")public String testValidNest(@RequestBody @Valid OuterValidatedData data) {System.out.println(data);return "OK";}@PostMapping("/validated/nest")public String testValidatedNest(@RequestBody @Validated OuterValidatedData data) {System.out.println(data);return "OK";}@PostMapping("/valid/method")public String testValidMethod(@RequestBody ValidatedData data) {validationService.testValid(data);return "OK";}@PostMapping("/validated/method")public String testValidatedMethod(@RequestBody ValidatedData data) {validationService.testValidated(data);return "OK";}@PostMapping("/valid/nest/method")public String testValidNestMethod(@RequestBody OuterValidatedData data) {validationService.testValidNest(data);return "OK";}@PostMapping("/validated/nest/method")public String testValidatedNestMethod(@RequestBody OuterValidatedData data) {validationService.testValidatedNest(data);return "OK";}@PostMapping("/valid/simple")public String testValid(@Valid @Max(10) int age, @Valid @NotBlank String name) {System.out.println(age + " " + name);return "OK";}@PostMapping("/validated/simple")public String testValidated(@Validated @Max(10) int age, @Validated @NotBlank String name) {System.out.println(age + " " + name);return "OK";}@PostMapping("/non/method/simple")public String testNonMethodSimple(@Max(10) int age, @NotBlank String name) {validationService.testNon(age, name);return "OK";}@PostMapping("/valid/method/simple")public String testValidMethodSimple(@Max(10) int age, @NotBlank String name) {validationService.testValid(age, name);return "OK";}@PostMapping("/validated/method/simple")public String testValidatedMethodSimple(@Max(10) int age, @NotBlank String name) {validationService.testValidated(age, name);return "OK";}
}
测试数据:
{"name": "demon","age": -1,"emails": ["demon7552003@hotmail.com"]
}{"name": "xxxxx","data": {"name": "demon","age": -1,"emails": ["demon7552003@hotmail.com"]}
}
测试服务
package com.jurassic.cloud.project.service.impl;import com.jurassic.cloud.project.dto.OuterValidatedData;
import com.jurassic.cloud.project.dto.ValidatedData;
import org.springframework.stereotype.Service;
import org.springframework.validation.annotation.Validated;import javax.validation.Valid;
import javax.validation.constraints.Max;
import javax.validation.constraints.NotBlank;/***** @author lihz* @date 2023/2/18*/
@Service
//@Validated
@Valid
public class ValidationService {public void testValid(@Valid ValidatedData data) {System.out.println(data);}public void testValidated(@Validated ValidatedData data) {System.out.println(data);}public void testValidNest(@Valid OuterValidatedData data) {System.out.println(data);}public void testValidatedNest(@Validated OuterValidatedData data) {System.out.println(data);}public void testNon( @Max(10) int age, @NotBlank String name) {System.out.println(age+" "+name);}public void testValid(@Valid @Max(10) int age,@Valid @NotBlank String name) {System.out.println(age+" "+name);}public void testValidated(@Validated @Max(10) int age, @Validated @NotBlank String name) {System.out.println(age+" "+name);}}
对JavaBean的校验(Controller层)
测试结果
-
valid测试{"code": 1,// age 必须是正数"msg": "Validation failed for argument [0] in public java.lang.String com.jurassic.cloud.project.controller.ValidationController.testValid(com.jurassic.cloud.project.dto.ValidatedData): [Field error in object 'validatedData' on field 'age': rejected value [-1]; codes [Positive.validatedData.age,Positive.age,Positive.java.lang.Integer,Positive]; arguments [org.springframework.context.support.DefaultMessageSourceResolvable: codes [validatedData.age,age]; arguments []; default message [age]]; default message [必须是正数]] ","data": null } -
validated测试
{"code": 1,"msg": "Validation failed for argument [0] in public java.lang.String com.jurassic.cloud.project.controller.ValidationController.testValidated(com.jurassic.cloud.project.dto.ValidatedData): [Field error in object 'validatedData' on field 'age': rejected value [-1]; codes [Positive.validatedData.age,Positive.age,Positive.java.lang.Integer,Positive]; arguments [org.springframework.context.support.DefaultMessageSourceResolvable: codes [validatedData.age,age]; arguments []; default message [age]]; default message [必须是正数]] ","data": null
}
-
valid测试 嵌套数据{"code": 1,"msg": "Validation failed for argument [0] in public java.lang.String com.jurassic.cloud.project.controller.ValidationController.testValidNest(com.jurassic.cloud.project.dto.OuterValidatedData): [Field error in object 'outerValidatedData' on field 'validatedData.age': rejected value [-1]; codes [Positive.outerValidatedData.validatedData.age,Positive.validatedData.age,Positive.age,Positive.java.lang.Integer,Positive]; arguments [org.springframework.context.support.DefaultMessageSourceResolvable: codes [outerValidatedData.validatedData.age,validatedData.age]; arguments []; default message [validatedData.age]]; default message [必须是正数]] ","data": null } -
validated测试 嵌套数据{"code": 1,"msg": "Validation failed for argument [0] in public java.lang.String com.jurassic.cloud.project.controller.ValidationController.testValidatedNest(com.jurassic.cloud.project.dto.OuterValidatedData): [Field error in object 'outerValidatedData' on field 'validatedData.age': rejected value [-1]; codes [Positive.outerValidatedData.validatedData.age,Positive.validatedData.age,Positive.age,Positive.java.lang.Integer,Positive]; arguments [org.springframework.context.support.DefaultMessageSourceResolvable: codes [outerValidatedData.validatedData.age,validatedData.age]; arguments []; default message [validatedData.age]]; default message [必须是正数]] ","data": null }
总结
从结果上看 @Valid 和 @Validated 都能触发级联验证。
如果要触发属性的级联验证,一定要放注解 @Valid。
对普通方法的JavaBean校验
测试结果
Service类上不加注解
-
valid测试OK -
validated测试ok -
valid测试 嵌套数据OK -
validated测试 嵌套数据OK
Service类上加@Valid
-
valid测试OK -
validated测试ok -
valid测试 嵌套数据OK -
validated测试 嵌套数据OK
Service类上加@Validated
-
valid测试{"code": 1,"msg": "testValid.data.age: 必须是正数","data": null } -
validated测试ok -
valid测试 嵌套数据{"code": 1,"msg": "testValidNest.data.validatedData.age: 必须是正数","data": null } -
validated测试 嵌套数据OK
总结
只有类上添加了@Vlidated注解,并且待校验的**JavaBean上添加了@Valid**的情况下校验才会生效。
返回的异常信息包含了 进行验证的方法的名称,以及验证的属性。
对简单参数的校验(Controller层)
测试结果
1、测试:http://127.0.0.1:8073/validate/valid/simple?age=20&name=demon
结果:OK
2、测试:http://127.0.0.1:8073/validate/validated/simple?age=20&name=demon
结果:OK
结论
参数上,不论加 @Valid 还是 @Validated ,都不会触发验证。
对简单参数的校验(Service层)
测试结果
1、测试:http://127.0.0.1:8073/validate/valid/simple?age=20&name=demon
结果:OK
2、测试:http://127.0.0.1:8073/validate/validated/simple?age=20&name=demon
结果:OK
结论
参数上,不论加 @Valid 还是 @Validated ,都不会触发验证。
对普通方法上的简单参数的校验(Service层)
1、参数不加注解: http://127.0.0.1:8073/validate/non/method/simple?age=20
2、参数加@Valid: http://127.0.0.1:8073/validate/valid/simple?age=20
3、参数加@Validated:http://127.0.0.1:8073/validate/validated/simple?age=20
测试结果
Service类上不加注解
1、OK
2、OK
3、OK
Service类上加@Valid
1、OK
2、OK
3、OK
Service类上加@Validated
1、
{"code": 1,"msg": "testNon.age: 最大不能超过10, testNon.name: 不能为空","data": null
}
2、
{"code": 1,"msg": "testValid.name: 不能为空, testValid.age: 最大不能超过10","data": null
}
3、
{"code": 1,"msg": "testValidated.age: 最大不能超过10, testValidated.name: 不能为空","data": null
}
结论
参数上,不论加 @Valid 还是 @Validated,或者不加 ,都不会影响结果。
仅在类上加 @Validated 时,才会触发简单参数的验证。
普通方法验证的总结
仅在类上加@Validated 时,才会触发参数的验证。对于非简单类的验证,必须加 @Valid 触发级联验证。
Controller层验证的总结
仅能验证非简单类型。
从结果上看 @Valid 和 @Validated 都能触发级联验证。
如果要触发属性的级联验证,一定要放注解 @Valid。
Spring对JSR的适配
在普通方法上的验证,抛出异常ConstraintViolationException,在Controller层抛出的异常是MethodArgumentNotValidException。Spring对参数绑定做了封装,错误信息被封装为BindingResult,在方法RequestResponseBodyMethodProcessor#resolveArgument中做处理。
全局异常处理
通过@RestControllerAdvice 做AOP。
@RestControllerAdvice
public class MethodArgumentNotValidExceptionHandler {@ExceptionHandler(MethodArgumentNotValidException.class)public Result handleMethodArgumentNotValid(MethodArgumentNotValidException ex) {BindingResult bindingResult = ex.getBindingResult();StringBuilder stringBuilder = new StringBuilder();for (FieldError error : bindingResult.getFieldErrors()) {String field = error.getField();Object value = error.getRejectedValue();String msg = error.getDefaultMessage();String message = String.format("错误字段:%s,错误值:%s,原因:%s;", field, value, msg);stringBuilder.append(message).append("\r\n");}return Result.error(MsgDefinition.ILLEGAL_ARGUMENTS.codeOf(), stringBuilder.toString());}
}
附录
参考
官网:https://docs.spring.io/spring-framework/docs/5.2.2.RELEASE/spring-framework-reference/core.html#validation-beanvalidation
相关文章:
Spring中的数据校验
数据校验基础 参考: Java Bean Validation 规范 Spring对Bean Validation的支持 Spring定义了一个接口org.springframework.validation.Validator,用于应用相关的对象的校验器。 这个接口完全从基础设施或者上下文中脱离的,这意味着它没有…...
python批量翻译excel表格中的英文
python批量翻译excel表格中的英文需求背景主要设计分析具体实现表格操作请求百度翻译api多线程控制台显示进度完整源码需求背景 女朋友的论文需要爬取YouTube视频热评,但爬下来的都是外文。 主要设计 读取一个表格文件,获取需要翻译的文本 使用百度翻译…...
基于SSM框架的RBAC权限系统设计与 实现
基于SSM框架的RBAC权限系统设计与 实现 ✌全网粉丝20W,csdn特邀作者、博客专家、CSDN新星计划导师、java领域优质创作者,博客之星、掘金/华为云/阿里云/InfoQ等平台优质作者、专注于Java技术领域和毕业项目实战✌ 🍅文末获取项目下载方式🍅 一、项目背景…...
目标检测各常见评价指标详解
注:本文仅供学习,未经同意请勿转载 说明:该博客来源于xiaobai_Ry:2020年3月笔记 对应的PDF下载链接在:待上传 目录 常见的评价指标 准确率 (Accuracy) 混淆矩阵 (Confusion Matrixÿ…...
深入讲解Kubernetes架构-控制器
在机器人技术和自动化领域,控制回路(Control Loop)是一个非终止回路,用于调节系统状态。这是一个控制环的例子:房间里的温度自动调节器。当你设置了温度,告诉了温度自动调节器你的期望状态(Desi…...
Urho3D本地化 国际化
本地化子系统提供了创建多语言应用程序的简单方法。 初始化 在使用子系统之前,需要加载本地化字符串集合。通常的做法是在应用程序启动时执行此操作。可以加载多个集合文件,每个集合文件只能定义一种或多种语言。例如: Localization* l10n…...
千锋教育嵌入式物联网教程之系统编程篇学习-04
目录 alarm函数 raise函数 abort函数 pause函数 转折点 signal函数 可重入函数 信号集 sigemptyset() sigfillset sigismember() sigaddset() sigdelset() 代码讲解 信号阻塞集 sigprocmask() alarm函数 相当于一个闹钟,默认动作是终止调用alarm函数的进…...
【运维】什么是 DevOps?
文章目录什么是 DevOps?如何实现 DevOpsDevOps工作原理: DevOps生命周期DevOps 文化DevOps 工具:构建 DevOps 工具链DevOps 和云原生开发什么是 DevSecOps?DevOps 和站点可靠性工程 (SRE)什么是 DevOps? DevOps 通过结…...
【C++入门】引用、内联函数、auto关键字、基于范围的for循环(C++11)、指针空值nullptr(C++11)
文章目录引用引用概念引用特性引用使用场景常引用内联函数宏的优缺点?C有哪些技术替代宏?auto关键字auto不能推导的场景基于范围的for循环(C11)指针空值nullptr(C11)引用 引用概念 引用不是新定义一个变量,而是给已存在变量取了一个别名&…...
《FPGA学习》->多个按键控制LED灯
🍎与其担心未来,不如现在好好努力。在这条路上,只有奋斗才能给你安全感。你若努力,全世界都会为你让路。本次项目任务,利用开发板上的4个按键KEY1,KEY2,KEY3,KEY4和2个LED灯LED1&…...
vb.net计算之.net core基础(4)-项目与程序结构(2)
目录 Namespace 语句Visual Basic 中的命名空间完全限定名命名空间可以定义什么全局关键字命名规范条件编译拆分和合并语句拆分成多行在同一行上放置多个语句为代码行添加标签注释串联成员访问运算符点运算符 `.`感叹号 `!`运算符Me 关键字MyMyBaseMyClassNamespace 语句 <…...
基于RK3588的嵌入式linux系统开发(五)——uboot优化修改(按任意按键停止autoboot)
我们通常情况下,芯片进入uboot后,会根据设置的bootdelay时间进行倒数计数。这时候在终端按任意键,即可退出autoboot,进入uboot的命令行模式。 官方提供的uboot源码中,为了防止调试串口干扰导致不能进入系统,…...
Lumerical---在FDTD和MODE工程中的PML边界条件
Lumerical---在FDTD和MODE工程中的PML边界条件 引言PML边界条件实现原理PML 类型PML 配置文件PML 配置文件选项Standard(标准)Stabilized(稳定性)Steep AngleCustom(陡角)对于不同的边界使用不同的配置FDE,varFDTD和FDTD SolverPML 参数阅读这篇前,推荐阅读边界条件综述…...
论文投稿指南——中文核心期刊推荐(社会学)
【前言】 🚀 想发论文怎么办?手把手教你论文如何投稿!那么,首先要搞懂投稿目标——论文期刊 🎄 在期刊论文的分布中,存在一种普遍现象:即对于某一特定的学科或专业来说,少数期刊所含…...
KVM-4、KVM 高级功能详解
1. 半虚拟化驱动 1.1 virtio 概述 KVM 是必须使用硬件虚拟化辅助技术(如 Intel VT-x 、AMD-V)的 Hypervisor,在CPU 运行效率方面有硬件支持,其效率是比较高的;在有 Intel EPT 特性支持的平台上,内存虚拟化的效率也较高。 QEMU/KVM 提供了全虚拟化环境,可以让客户机不经…...
【Linux】进程状态
文章目录1. 阻塞1. 举例2. 为什么要阻塞?3.操作系统层面上如何理解进程等待某种资源就绪呢?资源进程4. 总结2.挂起3.Linux进程状态1. R状态进程只要是R状态,就一定是在CPU运行吗?证明当前进程运行状态生成程序查看进程2. S休眠状态…...
2023河南省第二届职业技能大赛郑州市选拔赛“网络安全”项目比赛样题任务书
2023河南省第二届职业技能大赛郑州市选拔赛“网络安全” 项目比赛样题任务书 一、竞赛时间 共计360分钟。 竞赛任务书内容 2023河南省第二届职业技能大赛郑州市选拔赛“网络安全” 项目比赛样题任务书 A模块基础设施设置/安全加固(200分) A-1&…...
pygame8 扫雷游戏
一、游戏规则: 1、点击方格,如果是地雷,游戏失败,找到所有地雷游戏胜利 2、如果方块上出现数字,则表示在其周围的八个方块中共有多少颗地雷 二、游戏主逻辑: 主要逻辑即调用run_game, 然后循环检测事件…...
c/c++开发,无可避免的模板编程实践(篇四)
一、容器与模板 前文就说到,标准库基于模板编程,定义了许多容器类以及一系列泛型算法,使程序员可以更简洁、抽象和有效地编写程序。C标准库中有大量的标准容器,这些容器通常包含一组数据或对象的集合,几乎可以和任何类…...
c++11 标准模板(STL)(std::unordered_set)(二)
定义于头文件 <unordered_set> template< class Key, class Hash std::hash<Key>, class KeyEqual std::equal_to<Key>, class Allocator std::allocator<Key> > class unordered_set;(1)(C11 起)namespace pmr { templ…...
将对透视变换后的图像使用Otsu进行阈值化,来分离黑色和白色像素。这句话中的Otsu是什么意思?
Otsu 是一种自动阈值化方法,用于将图像分割为前景和背景。它通过最小化图像的类内方差或等价地最大化类间方差来选择最佳阈值。这种方法特别适用于图像的二值化处理,能够自动确定一个阈值,将图像中的像素分为黑色和白色两类。 Otsu 方法的原…...
Frozen-Flask :将 Flask 应用“冻结”为静态文件
Frozen-Flask 是一个用于将 Flask 应用“冻结”为静态文件的 Python 扩展。它的核心用途是:将一个 Flask Web 应用生成成纯静态 HTML 文件,从而可以部署到静态网站托管服务上,如 GitHub Pages、Netlify 或任何支持静态文件的网站服务器。 &am…...
屋顶变身“发电站” ,中天合创屋面分布式光伏发电项目顺利并网!
5月28日,中天合创屋面分布式光伏发电项目顺利并网发电,该项目位于内蒙古自治区鄂尔多斯市乌审旗,项目利用中天合创聚乙烯、聚丙烯仓库屋面作为场地建设光伏电站,总装机容量为9.96MWp。 项目投运后,每年可节约标煤3670…...
【论文笔记】若干矿井粉尘检测算法概述
总的来说,传统机器学习、传统机器学习与深度学习的结合、LSTM等算法所需要的数据集来源于矿井传感器测量的粉尘浓度,通过建立回归模型来预测未来矿井的粉尘浓度。传统机器学习算法性能易受数据中极端值的影响。YOLO等计算机视觉算法所需要的数据集来源于…...
第一篇:Agent2Agent (A2A) 协议——协作式人工智能的黎明
AI 领域的快速发展正在催生一个新时代,智能代理(agents)不再是孤立的个体,而是能够像一个数字团队一样协作。然而,当前 AI 生态系统的碎片化阻碍了这一愿景的实现,导致了“AI 巴别塔问题”——不同代理之间…...
12.找到字符串中所有字母异位词
🧠 题目解析 题目描述: 给定两个字符串 s 和 p,找出 s 中所有 p 的字母异位词的起始索引。 返回的答案以数组形式表示。 字母异位词定义: 若两个字符串包含的字符种类和出现次数完全相同,顺序无所谓,则互为…...
鸿蒙DevEco Studio HarmonyOS 5跑酷小游戏实现指南
1. 项目概述 本跑酷小游戏基于鸿蒙HarmonyOS 5开发,使用DevEco Studio作为开发工具,采用Java语言实现,包含角色控制、障碍物生成和分数计算系统。 2. 项目结构 /src/main/java/com/example/runner/├── MainAbilitySlice.java // 主界…...
Redis的发布订阅模式与专业的 MQ(如 Kafka, RabbitMQ)相比,优缺点是什么?适用于哪些场景?
Redis 的发布订阅(Pub/Sub)模式与专业的 MQ(Message Queue)如 Kafka、RabbitMQ 进行比较,核心的权衡点在于:简单与速度 vs. 可靠与功能。 下面我们详细展开对比。 Redis Pub/Sub 的核心特点 它是一个发后…...
2025季度云服务器排行榜
在全球云服务器市场,各厂商的排名和地位并非一成不变,而是由其独特的优势、战略布局和市场适应性共同决定的。以下是根据2025年市场趋势,对主要云服务器厂商在排行榜中占据重要位置的原因和优势进行深度分析: 一、全球“三巨头”…...
C++.OpenGL (20/64)混合(Blending)
混合(Blending) 透明效果核心原理 #mermaid-svg-SWG0UzVfJms7Sm3e {font-family:"trebuchet ms",verdana,arial,sans-serif;font-size:16px;fill:#333;}#mermaid-svg-SWG0UzVfJms7Sm3e .error-icon{fill:#552222;}#mermaid-svg-SWG0UzVfJms7Sm3e .error-text{fill…...
