当请求参数很多,几乎每一个参数都需要后端去兜底校验时,你还在写if else去判断参数是否为空吗??要校验为空的参数三四个还好,要是十几个,业务逻辑还没开始就写二三十行代码开始堆山了嘛,教给大家一个简单实用的方法,让大家不再在屎堆上堆代码。
先来了解几个常用的注解
| 注解 | 含义 | 怎样使用 |
|---|---|---|
| @NotBlank | 字符串不为null且非空格长度至少为1 | 注解在String类型的参数上 |
| @Null | 必须为null | 用于Long、Integer、BigDecimal基本数据类型上 |
| @NotNull | 必须不为null | 用于Long、Integer、BigDecimal基本数据类型上 |
| @NotEmpty | 集合不为null且长度>0 | 用于集合校验 |
| @AssertTrue | 为true | 用于Boolean类型上 |
| @AssertFalse | 为false | 用于Boolean类型上 |
| @Max | 小于或等于某个数 | 一个字段的最大值为xx |
| @Min | 大于或等于某个数 | 一个字段的最小值为xx |
| @Digits(integer=3, fraction=2) | 整数和小数长度 | — |
| @Pattern | 正则匹配 | 手机号规则校验等 |
| @Range(min=,max=) | 数值类型范围 | — |
| @Length | 字符串长度范围 | ---- |
BizRequestDTO
import lombok.Data;
import org.hibernate.validator.constraints.NotBlank;
import javax.validation.Valid;
import java.util.Date;
import java.util.List;
@Data
public class BizRequestDTO {
/**
* 单据编号
*/
@NotBlank(message = "headId (单据编号)不能为空")
private String headId;
/**
* 金额
*/
@NotNull(message = "amount (金额)不能为空")
private BigDecimal amount;
/**
* 提单人
*/
@Valid
private EmpInfoDTO submitter;
/**
* 分摊人列表
*/
@Valid
private List<EmpInfoDTO> expenseSharerList;
}
作用于对象,作用于集合 对象内校验方式:
@Valid
private EmpInfoDTO submitter;
EmpInfoDTO
/**
* 工号
*/
@NotBlank(message = "工号不能为空")
private String workNo;
/**
* 姓名
*/
@NotBlank(message = "姓名不能为空")
private String nickName;
引入依赖
<dependency>
<groupId>org.hibernate</groupId>
<artifactId>hibernate-validator</artifactId>
<version>6.0.16.Final</version>
</dependency>
<dependency>
<groupId>javax.validation</groupId>
<artifactId>validation-api</artifactId>
<version>2.0.1.Final</version>
</dependency>
引入一个工具类
import org.hibernate.validator.HibernateValidator;
import javax.validation.ConstraintViolation;
import javax.validation.Validation;
import javax.validation.Validator;
import java.util.Set;
public class ValidatorUtils {
private static final Validator DEFAULT_VALIDATOR;
static {
DEFAULT_VALIDATOR = Validation.byProvider(HibernateValidator.class)
.configure()
.buildValidatorFactory()
.getValidator();
}
public static <T> Set<ConstraintViolation<T>> validate(T bean) {
return DEFAULT_VALIDATOR.validate(bean);
}
}
实际应用
public ResponseResult<String> checkExpenseParams(RequestParamDTO requestDTO) {
Set<ConstraintViolation<TaeExpenseLineCheckRequestDTO>> result = ValidatorUtils.validate(requestDTO);
// 错误信息输出
StringBuilder resultInfo = new StringBuilder();
result.forEach(validateResult -> resultInfo.append(validateResult.getMessage()).append("\n"));
if (StringUtils.isNotBlank(resultInfo)) {
return ResponseResult.error(resultInfo.toString());
}
// 若为空则参数校验通过
return ResponseResult.ok(null);
}
输出示例
headId (单据编号)不能为空
amount (金额)不能为空
Hibernate可以对校验模式的配置,支持快速失败,即在校验的过程中只要有一项不同过,剩余的参数就不再进行校验。默认使用的是全部校验完成后再返回,也就是在上面的例子中展示的那样。开启快速失败的具体的实现方式:
Validation.byProvider(HibernateValidator.class)
.configure()
.failFast(true)
.buildValidatorFactory()
.getValidator();
校验工具类
import org.hibernate.validator.HibernateValidator;
import javax.validation.ConstraintViolation;
import javax.validation.Validator;
import javax.validation.Validation;
import java.util.Set;
public class ValidatorUtils {
private static final Validator DEFAULT_VALIDATOR;
private static final Validator FAST_VALIDATOR;
static {
DEFAULT_VALIDATOR = Validation.byProvider(HibernateValidator.class)
.configure()
.buildValidatorFactory()
.getValidator();
FAST_VALIDATOR = Validation.byProvider(HibernateValidator.class)
.configure()
.failFast(true)
.buildValidatorFactory()
.getValidator();
}
public static <T> Set<ConstraintViolation<T>> validate(T bean) {
return DEFAULT_VALIDATOR.validate(bean);
}
public static <T> Set<ConstraintViolation<T>> fastValidate(T bean) {
return FAST_VALIDATOR.validate(bean);
}
}
快速校验使用方式同上,输出示例
headId (单据编号)不能为空
包含级联对象的类
对象级联校验意味着,在一个对象中的属性可以包含另外一个校验对象,被级联的对象需要用@Valid注解修饰。上面也有给出🌰,本节再详细讲解一下。
例:
import lombok.Data;
import org.hibernate.validator.constraints.NotBlank;
import org.hibernate.validator.constraints.Range;
import javax.validation.Valid;
import javax.validation.constraints.NotNull;
import javax.validation.constraints.Pattern;
@Data
public class ValidatorWithDefineExtraModel {
@NotBlank(message="姓名不能为空")
private String name;
@NotNull(message = "年龄不能为空")
@Range(min=1, max=200, message = "年龄必须大于1小于200")
private Integer age;
@NotBlank(message="性别不能为空")
@Pattern(regexp = "([FM])", message = "性别只能为F(女)或者M(男)")
private String sex;
@NotNull(message = "validatorModelExtra不能为空")
@Valid
private ValidatorModelExtra validatorModelExtra;
}
被级联的对象
@Data
public class ValidatorModelExtra {
@NotNull(message = "地址不能为空")
@Length(max = 100, message = "地址长度不能大于100个字符")
private String address;
}
输出示例:
地址不能为空
很好奇,就使用rubyonrails自动化单元测试而言,你们正在做什么?您是否创建了一个脚本来在cron中运行rake作业并将结果邮寄给您?git中的预提交Hook?只是手动调用?我完全理解测试,但想知道在错误发生之前捕获错误的最佳实践是什么。让我们理所当然地认为测试本身是完美无缺的,并且可以正常工作。下一步是什么以确保他们在正确的时间将可能有害的结果传达给您? 最佳答案 不确定您到底想听什么,但是有几个级别的自动代码库控制:在处理某项功能时,您可以使用类似autotest的内容获得关于哪些有效,哪些无效的即时反馈。要确保您的提
在控制台中反复尝试之后,我想到了这种方法,可以按发生日期对类似activerecord的(Mongoid)对象进行分组。我不确定这是完成此任务的最佳方法,但它确实有效。有没有人有更好的建议,或者这是一个很好的方法?#eventsisanarrayofactiverecord-likeobjectsthatincludeatimeattributeevents.map{|event|#converteventsarrayintoanarrayofhasheswiththedayofthemonthandtheevent{:number=>event.time.day,:event=>ev
我正在编写一个包含C扩展的gem。通常当我写一个gem时,我会遵循TDD的过程,我会写一个失败的规范,然后处理代码直到它通过,等等......在“ext/mygem/mygem.c”中我的C扩展和在gemspec的“扩展”中配置的有效extconf.rb,如何运行我的规范并仍然加载我的C扩展?当我更改C代码时,我需要采取哪些步骤来重新编译代码?这可能是个愚蠢的问题,但是从我的gem的开发源代码树中输入“bundleinstall”不会构建任何native扩展。当我手动运行rubyext/mygem/extconf.rb时,我确实得到了一个Makefile(在整个项目的根目录中),然后当
exe应该在我打开页面时运行。异步进程需要运行。有什么方法可以在ruby中使用两个参数异步运行exe吗?我已经尝试过ruby命令-system()、exec()但它正在等待过程完成。我需要用参数启动exe,无需等待进程完成是否有任何rubygems会支持我的问题? 最佳答案 您可以使用Process.spawn和Process.wait2:pid=Process.spawn'your.exe','--option'#Later...pid,status=Process.wait2pid您的程序将作为解释器的子进程执行。除
设置:狂欢ruby1.9.2高线(1.6.13)描述:我已经相当习惯在其他一些项目中使用highline,但已经有几个月没有使用它了。现在,在Ruby1.9.2上全新安装时,它似乎不允许在同一行回答提示。所以以前我会看到类似的东西:require"highline/import"ask"Whatisyourfavoritecolor?"并得到:Whatisyourfavoritecolor?|现在我看到类似的东西:Whatisyourfavoritecolor?|竖线(|)符号是我的终端光标。知道为什么会发生这种变化吗? 最佳答案
我有一些Ruby代码,如下所示:Something.createdo|x|x.foo=barend我想编写一个测试,它使用double代替block参数x,这样我就可以调用:x_double.should_receive(:foo).with("whatever").这可能吗? 最佳答案 specify'something'dox=doublex.should_receive(:foo=).with("whatever")Something.should_receive(:create).and_yield(x)#callthere
我正在为一个项目制作一个简单的shell,我希望像在Bash中一样解析参数字符串。foobar"helloworld"fooz应该变成:["foo","bar","helloworld","fooz"]等等。到目前为止,我一直在使用CSV::parse_line,将列分隔符设置为""和.compact输出。问题是我现在必须选择是要支持单引号还是双引号。CSV不支持超过一个分隔符。Python有一个名为shlex的模块:>>>shlex.split("Test'helloworld'foo")['Test','helloworld','foo']>>>shlex.split('Test"
我在我的Rails项目中使用Pow和powifygem。现在我尝试升级我的ruby版本(从1.9.3到2.0.0,我使用RVM)当我切换ruby版本、安装所有gem依赖项时,我通过运行railss并访问localhost:3000确保该应用程序正常运行以前,我通过使用pow访问http://my_app.dev来浏览我的应用程序。升级后,由于错误Bundler::RubyVersionMismatch:YourRubyversionis1.9.3,butyourGemfilespecified2.0.0,此url不起作用我尝试过的:重新创建pow应用程序重启pow服务器更新战俘
我不确定传递给方法的对象的类型是否正确。我可能会将一个字符串传递给一个只能处理整数的函数。某种运行时保证怎么样?我看不到比以下更好的选择:defsomeFixNumMangler(input)raise"wrongtype:integerrequired"unlessinput.class==FixNumother_stuffend有更好的选择吗? 最佳答案 使用Kernel#Integer在使用之前转换输入的方法。当无法以任何合理的方式将输入转换为整数时,它将引发ArgumentError。defmy_method(number)
我真的很习惯使用Ruby编写以下代码:my_hash={}my_hash['test']=1Java中对应的数据结构是什么? 最佳答案 HashMapmap=newHashMap();map.put("test",1);我假设? 关于java-等价于Java中的RubyHash,我们在StackOverflow上找到一个类似的问题: https://stackoverflow.com/questions/22737685/