- iOS/Objective-C 元类和类别
- objective-c - -1001 错误,当 NSURLSession 通过 httpproxy 和/etc/hosts
- java - 使用网络类获取 url 地址
- ios - 推送通知中不播放声音
我是 Spring MVC 的新手,我导入了一个与服务器端验证相关的教程项目,但我对它的工作原理有些怀疑。
所以我有一个名为 login.jsp 的登录页面,其中包含此登录表单:
<form:form action="${pageContext.request.contextPath}/login" commandName="user" method="post">
<table>
<tr>
<td><label>Enter Username : </label></td>
<td><form:input type="text" path="username" name="username" />
<br> <form:errors path="username" style="color:red;"></form:errors>
</td>
</tr>
<tr>
<td><label>Enter Password : </label></td>
<td><form:input type="password" path="password" name="password" />
<br> <form:errors path="password" style="color:red;"></form:errors>
</td>
</tr>
<tr>
<td> </td>
<td align="center"><input type="submit" value="Login" /></td>
</tr>
</table>
</form:form>
我认为使用从模型中检索到的 commandName="user" 属性指定的对象(如果我错了请纠正我)来存储用户输入的用户名和密码。
这个commandName="user" 是这个User 类的实例:
import javax.validation.constraints.Size;
import org.hibernate.validator.constraints.NotBlank;
public class User {
@NotBlank(message="Username can not be blank")
private String username;
@Size(min=6,message="Password must be atleast 6 characters long")
private String password;
private String gender;
private String vehicle;
private String country;
private String image;
...............................................
...............................................
GETTER AND SETTER METHODS
...............................................
...............................................
}
如您所见,@NotBlank 和 @Size 验证注解在用户名 和密码 字段。
这里第一个疑问:与 2 个使用的库 javax.validation 和 org.hibernate.validator 到底有什么区别?
为什么在教程中同时使用两者?我可以只使用 hibernate 验证程序库做同样的事情吗? (我认为我是否可以使用 Hibernate validator 指定字符串的有效长度)?
因此,当提交登录表单时,它会生成 HttpRequest 并向 /login 资源发出请求,该资源由声明到 Controller 类中的此方法处理:
@RequestMapping(value="/login" , method=RequestMethod.POST)
public String do_login(HttpServletRequest req , Model md , HttpSession session , @Valid User user, BindingResult br)
{
try
{
//System.out.println(br.getAllErrors().size());
String username = req.getParameter("username");
String password = req.getParameter("password");
System.out.println("Username and pasword are : "+username +" "+ password);
if(br.getAllErrors().size() > 0){
System.out.println("Server side validation takes place....");
}
else{
Login_Model lm = new Login_Model();
String message = lm.do_login_process(username, password);
if(message.equals("login success"))
{
session.setAttribute("username", username);
return "redirect:/myprofile";
}
else
{
md.addAttribute("error_msg", message);
}
}
return "login";
}
catch(Exception e)
{
return "login";
}
}
OK,现在我对这个方法有如下疑惑:
将此对象作为输入参数:@Valid User 用户。谁传给它?我认为这可能取决于我在表单中指定 commandName="user" 的事实,因此 Spring 会自动执行此操作。是否正确?
据我了解,@Valid 注释会自动调用验证过程。它是怎么发生的?与Spring 提供的AOP 特性有关吗?或者是什么?为什么这个 @Valid 注释只与 javax.validation 库相关,而不与 Hibernate validator 相关(所以这个 @Valid 注释也验证用 Hibernate validator 注释注释的字段?为什么?)
据我了解,如果用户在登录表单中插入错误的值,我可以通过 BindingResult br 输入参数获取此错误。使用调试器,我可以看到该对象包含由定义到 User 模型对象中的注释定义的错误消息。这究竟是如何工作的?
最佳答案
And here the first doubt: what exactly is the difference between the 2 used libraries
javax.validation
andorg.hibernate.validator
?
javax.validation
来自 JSR-303 API。您可以查看 API 类,例如 this Maven dependency .
Hibernate validator是 JSR 303 的实现之一(实际上是引用实现),因此它实现了所有 API,但添加了自己的扩展,例如 @NotBlank
您提到的注释。 JSR 303 的其他实现是例如 Apache BVal .
It takes this object as input parameter:
@Valid User user
. Who passes it to the handler method?
Spring MVC 中处理程序方法的值由接口(interface) HandlerMethodArgumentResolver
的实现提供.在您的情况下,将调用以解析 User
参数的实现可能是 ServletModelAttributeMethodProcessor
.您可以查看这些类的源代码和 JavaDocs 以了解它们的内部工作方式。
The
@Valid
annotation will automatically call the validation process. How does this happend? Does it use AOP? Why this@Valid
annotation is related only to thejavax.validation
library and not also to the Hibernate validator (so this@Valid
annotation validate also the field annotated with Hibernate validator annotation?)
验证过程由 ModelAttributeMethodProcessor
调用,也就是前面提到的class ServletModelAttributeMethodProcessor
继承自。它包含以下方法:
protected void validateIfApplicable(WebDataBinder binder, MethodParameter parameter) {
Annotation[] annotations = parameter.getParameterAnnotations();
for (Annotation ann : annotations) {
if (ann.annotationType().getSimpleName().startsWith("Valid")) {
Object hints = AnnotationUtils.getValue(ann);
binder.validate(hints instanceof Object[] ? (Object[]) hints : new Object[] {hints});
break;
}
}
}
如果仔细观察,您会看到以下表达式的条件:
ann.annotationType().getSimpleName().startsWith("Valid")
这意味着如果参数具有以Valid
开头的任何注释,Spring 将调用验证。它可能是 JSR 303 的 @Valid
或 Spring 的 @Validated
它支持验证组。它甚至可以是您的自定义注释,只要其名称以 Valid
开头即可。
From what I have understand if the user inserts wrong values into the login form I can obtain this error from the
BindingResult
handler parameter. Using the debugger I can see that this object contain the error message defined by the annotation defined into the User model object. How exactly works?
让我们回到 ModelAttributeMethodProcessor
类。在其resolveArgument
方法中有如下代码:
WebDataBinder binder = binderFactory.createBinder(webRequest, attribute, name);
这将创建 WebDataBinder
的实例,在前面提到的 validateIfApplicable
方法中调用验证。验证本身填充 BindingResult
,然后通过 ErrorsMethodArgumentResolver
类提供给 Controller 处理程序方法,该类再次实现 HandlerMethodArgumentResolver
。
TLDR:您在这个问题中提出的很多问题都可以追溯到 HandlerMethodArgumentResolver
的各种实现。我建议浏览这些类并使用调试器逐步浏览它们以更好地理解它们。
关于java - Spring MVC 如何解析和验证处理程序方法参数?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/33053553/
我一直在使用 AJAX 从我正在创建的网络服务中解析 JSON 数组时遇到问题。我的前端是一个简单的 ajax 和 jquery 组合,用于显示从我正在创建的网络服务返回的结果。 尽管知道我的数据库查
很难说出这里要问什么。这个问题模棱两可、含糊不清、不完整、过于宽泛或夸夸其谈,无法以目前的形式得到合理的回答。如需帮助澄清此问题以便重新打开,visit the help center . 关闭 1
我在尝试运行 Android 应用程序时遇到问题并收到以下错误 java.lang.NoClassDefFoundError: com.parse.Parse 当我尝试运行该应用时。 最佳答案 在这
有什么办法可以防止etree在解析HTML内容时解析HTML实体吗? html = etree.HTML('&') html.find('.//body').text 这给了我 '&' 但我想
我有一个有点疯狂的例子,但对于那些 JavaScript 函数作用域专家来说,它看起来是一个很好的练习: (function (global) { // our module number one
关闭。此题需要details or clarity 。目前不接受答案。 想要改进这个问题吗?通过 editing this post 添加详细信息并澄清问题. 已关闭 8 年前。 Improve th
我需要编写一个脚本来获取链接并解析链接页面的 HTML 以提取标题和其他一些数据,例如可能是简短的描述,就像您链接到 Facebook 上的内容一样。 当用户向站点添加链接时将调用它,因此在客户端启动
在 VS Code 中本地开发时,包解析为 C:/Users//AppData/Local/Microsoft/TypeScript/3.5/node_modules/@types//index而不是
我在将 json 从 php 解析为 javascript 时遇到问题 这是我的示例代码: //function MethodAjax = function (wsFile, param) {
我在将 json 从 php 解析为 javascript 时遇到问题 这是我的示例代码: //function MethodAjax = function (wsFile, param) {
我被赋予了将一种语言“翻译”成另一种语言的工作。对于使用正则表达式的简单逐行方法来说,源代码过于灵活(复杂)。我在哪里可以了解更多关于词法分析和解析器的信息? 最佳答案 如果你想对这个主题产生“情绪化
您好,我在解析此文本时遇到问题 { { { {[system1];1;1;0.612509325}; {[system2];1;
我正在为 adobe after effects 在 extendscript 中编写一些代码,最终变成了 javascript。 我有一个数组,我想只搜索单词“assemble”并返回整个 jc3_
我有这段代码: $(document).ready(function() { // }); 问题:FB_RequireFeatures block 外部的代码先于其内部的代码执行。因此 who
背景: netcore项目中有些服务是在通过中间件来通信的,比如orleans组件。它里面服务和客户端会指定网关和端口,我们只需要开放客户端给外界,服务端关闭端口。相当于去掉host,这样省掉了些
1.首先贴上我试验成功的代码 复制代码 代码如下: protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec)
什么是 XML? XML 指可扩展标记语言(eXtensible Markup Language),标准通用标记语言的子集,是一种用于标记电子文件使其具有结构性的标记语言。 你可以通过本站学习 X
【PHP代码】 复制代码 代码如下: $stmt = mssql_init('P__Global_Test', $conn) or die("initialize sto
在SQL查询分析器执行以下代码就可以了。 复制代码代码如下: declare @t varchar(255),@c varchar(255) declare table_cursor curs
前言 最近练习了一些前端算法题,现在做个总结,以下题目都是个人写法,并不是标准答案,如有错误欢迎指出,有对某道题有新的想法的友友也可以在评论区发表想法,互相学习🤭 题目 题目一: 二维数组中的
我是一名优秀的程序员,十分优秀!