Spring MVC框架:第十四章:數(shù)據(jù)校驗(yàn)

第十二章 數(shù)據(jù)校驗(yàn)

在Web應(yīng)用三層架構(gòu)體系中,表述層負(fù)責(zé)接收瀏覽器提交的數(shù)據(jù),業(yè)務(wù)邏輯層負(fù)責(zé)數(shù)據(jù)的處理。為了能夠讓業(yè)務(wù)邏輯層基于正確的數(shù)據(jù)進(jìn)行處理,我們需要在表述層對(duì)數(shù)據(jù)進(jìn)行檢查,將錯(cuò)誤的數(shù)據(jù)隔絕在業(yè)務(wù)邏輯層之外。

1.校驗(yàn)概述

JSR 303是Java為Bean數(shù)據(jù)合法性校驗(yàn)提供的標(biāo)準(zhǔn)框架,它已經(jīng)包含在JavaEE 6.0標(biāo)準(zhǔn)中。JSR 303通過(guò)在Bean 屬性上標(biāo)注類似于@NotNull、@Max等標(biāo)準(zhǔn)的注解指定校驗(yàn)規(guī)則,并通過(guò)標(biāo)準(zhǔn)的驗(yàn)證接口對(duì)Bean進(jìn)行驗(yàn)證。

在這里插入圖片描述
JSR 303只是一套標(biāo)準(zhǔn),需要提供其實(shí)現(xiàn)才可以使用。Hibernate Validator是JSR 303的一個(gè)參考實(shí)現(xiàn),除支持所有標(biāo)準(zhǔn)的校驗(yàn)注解外,它還支持以下的擴(kuò)展注解:
在這里插入圖片描述
Spring4.0擁有自己獨(dú)立的數(shù)據(jù)校驗(yàn)框架,同時(shí)支持JSR 303標(biāo)準(zhǔn)的校驗(yàn)框架。Spring在進(jìn)行數(shù)據(jù)綁定時(shí),可同時(shí)調(diào)用校驗(yàn)框架完成數(shù)據(jù)校驗(yàn)工作。在Spring MVC中,可直接通過(guò)注解驅(qū)動(dòng)(mvc:annotation-driven)的方式進(jìn)行數(shù)據(jù)校驗(yàn)。Spring的LocalValidatorFactroyBean既實(shí)現(xiàn)了Spring的Validator接口,也實(shí)現(xiàn)了JSR 303的 Validator接口。只要在Spring容器中定義了一個(gè)LocalValidatorFactoryBean,即可將其注入到需要數(shù)據(jù)校驗(yàn)的 Bean中。Spring本身并沒(méi)有提供JSR 303的實(shí)現(xiàn),所以必須將JSR 303的實(shí)現(xiàn)者的jar包放到類路徑下。

配置mvc:annotation-driven/后,SpringMVC會(huì)默認(rèn)裝配好一個(gè)LocalValidatorFactoryBean,通過(guò)在處理方法的入?yún)⑸蠘?biāo)注@Validated注解即可讓Spring MVC在完成數(shù)據(jù)綁定后執(zhí)行數(shù)據(jù)校驗(yàn)的工作。

2.操作步驟
①導(dǎo)入驗(yàn)證環(huán)境

[1]在當(dāng)前工程類路徑下加入以下jar包

1classmate-0.8.0.jar

2hibernate-validator-5.0.0.CR2.jar

3hibernate-validator-annotation-processor-5.0.0.CR2.jar

4jboss-logging-3.1.1.GA.jar

5validation-api-1.1.0.CR1.jar

[2]在Tomcat的lib目錄下加入以下jar包[Tomcat7及以上版本不需要加]

1javax.el-2.2.4.jar

2javax.el-api-2.2.4.jar

3el-api-2.2.jar

②在需要驗(yàn)證的字段上添加驗(yàn)證注解

@NotEmpty
private String empName;

③目標(biāo)方法的bean前添加@Validated注解

※注意:在@Validated注解修飾的bean之后, 緊跟Errors(或BindingResult)類型的參數(shù)。在@Validated注解修飾的bean和Errors參數(shù)之間不能再有其他的參數(shù)!

④調(diào)用hasErrors()方法在目標(biāo)方法中檢驗(yàn)是否沒(méi)有通過(guò)驗(yàn)證

	 //檢測(cè)是否存在“數(shù)據(jù)綁定”錯(cuò)誤
	boolean hasErrors = bindingResult.hasErrors();
	if(hasErrors) {
		return "error";
	}
  • 1
  • 2
  • 3
  • 4
  • 5

⑤顯示錯(cuò)誤消息

[1]使form:errors標(biāo)簽 [2]form:errors 標(biāo)簽必須在 SpringMVC 的 form:form 標(biāo)簽中使用

<form:form modelAttribute=“employee”>
<form:input path=“age”/><form:errors path=“age”/>
</form:form>

實(shí)戰(zhàn)案例:
正確的數(shù)據(jù)
頁(yè)面:

測(cè)試自動(dòng)的類型轉(zhuǎn)換

1
  • 1

Employee

public class Employee {

private Integer age;

@DateTimeFormat(pattern="yyyy-MM-dd")
private Date birthday;
  • 1
  • 2
  • 3
  • 4

spring-mvc.xml

<context:component-scan base-package=“com.spring.mvc.handlers”/>

<bean id="viewResolver" class="org.springframework.web.servlet.view.InternalResourceViewResolver">
	<property name="prefix" value="/WEB-INF/page/"/>
	<property name="suffix" value=".jsp"/>
</bean>
  • 1
  • 2
  • 3
  • 4

handlers

@RequestMapping("/convert")
public String convertFail(Employee employee, BindingResult bindingResult, Model model) {
System.out.println(employee);
//檢測(cè)是否存在“數(shù)據(jù)綁定”錯(cuò)誤
boolean hasErrors = bindingResult.hasErrors();
if(hasErrors) {
return “error”;
}
model.addAttribute(“employee”, employee);
return “target”;
}

error.jsp

Error

target.jsp

Target

頁(yè)面點(diǎn)擊‘’測(cè)試自動(dòng)的類型轉(zhuǎn)換‘’
控制臺(tái)輸出Employee [age=15, birthday=null]
在這里插入圖片描述






給一個(gè)類型錯(cuò)誤的數(shù)據(jù)
頁(yè)面:

測(cè)試自動(dòng)的類型轉(zhuǎn)換

1
  • 1

Employee

public class Employee {

private Integer age;
  • 1

點(diǎn)擊頁(yè)面‘’測(cè)試自動(dòng)的類型轉(zhuǎn)換‘’后
控制臺(tái)輸出Employee [age=null, birthday=null]

頁(yè)面顯示
在這里插入圖片描述




我們?cè)偕钊肓私鈹?shù)據(jù)校驗(yàn),自定義類型轉(zhuǎn)換
直接上代碼更直觀
還是引用上面的代碼
spring-mvc.xml

<context:component-scan base-package=“com.spring.mvc.handlers”/>

<bean id="viewResolver" class="org.springframework.web.servlet.view.InternalResourceViewResolver">
	<property name="prefix" value="/WEB-INF/page/"/>
	<property name="suffix" value=".jsp"/>
</bean>
  • 1
  • 2
  • 3
  • 4

Address

public class Address {

private String city;
private String street;
private String no;
  • 1
  • 2
  • 3

Student

public class Student {

private Integer stuId;
private String stuName;

private Address address;
  • 1
  • 2
  • 3
  • 4

AddressConvert

import org.springframework.core.convert.converter.Converter;
import com.atguigu.spring.mvc.entity.Address;

public class AddressConvert implements Converter<String, Address> {

@Override
public Address convert(String source) {
	//使用逗號(hào)分隔符拆分字符串
	String[] split = source.split(",");
	//從拆分得到的數(shù)組中獲取Address對(duì)象的各個(gè)屬性值
	String city = split[0];
	String street = split[1];
	String no = split[2];
	//創(chuàng)建對(duì)象并返回
	return new Address(city, street, no);
}
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11

}

頁(yè)面:

id:
姓名:
地址:

handlers

@Controller
public class ConvertHandler {

@RequestMapping("/convert/save/stu")
public String saveStu(Student student) {
	System.out.println(student);
	return "target";
}
  • 1
  • 2
  • 3
  • 4
  • 5

頁(yè)面
在這里插入圖片描述






控制臺(tái)輸出
Student [stuId=45, stuName=gf, address=Address [city=fgf, street=hfdfd, no=gfddf]]

如果Student和Address實(shí)體類沒(méi)有給toString方法,那address=只有地址

執(zhí)行saveStu方法完
在這里插入圖片描述