七、Spring MVC高级技术

知识点

  • 处理文件上传
  • 使用flash属性
  • 在控制器中处理异常

关键词

  • 控制器通知 (Controller Advice)

7.1 处理异常

Spring提供了多种方式将异常转换为响应:

  • 特定的Spring异常将会自动映射为指定的HTTP状态码
  • 异常上可以添加@RequestStatus注解,从而将其映射为某一个HTTP状态码
  • 在方法上可以添加@ExceptionHandler注解,使其用来处理异常

最简单的方式就是第二种,映射。

7.1.1 将异常映射为HTTP状态码

在默认情况下,Spring会将自身的一些异常自动转化为合适的状态码。

Spring异常 HTTP状态码
BindException 400-Bad Request
ConversionNotSupportedException 500-Internal Server Error
HttpMessageNotReadableException 400-Bad Request
HttpMessageNotWriteableException 500-Internal Server Error
HttpRequestMethodNotSupportedException 405-Method Not Allowed
MethodArgumentNotValidException 400-Bad Request
MissingServletRequestParameterException 400-Bad Request
MissingServletRequestPartException 400-Bad Request
NoSuchRequestHandlingMethodException 404-Not Found
TypeMismatchException 400-Bad Request
@ResponseStatus(value=HttpStatus.NOT_FOUND, reason="Spittle Not Found")
public class .....

7.2 为控制器添加通知

控制器通知是任意带有@ControllerAdvice注解的类,这个类包含以下的方法:

  • @ExceptionHandler 注解标注的fangfa
  • @InitBinder注解标注的方法
  • @ModelAttribute注解标注的方法
    以上这些方法会自动运用到整个应用程序所有控制器中带有@RequestMapping注解的方法上。
// 写一个控制器异常处理类
package test
import org.springframework.web.bind.annotation.ControllerAdvice;
import org.springframework.web.bind.annotation.ExceptionHandler;
// 定义控制器类
@ControllerAdvice
public class AppWideExceptionHandler {
	// 定义处理异常的方法
	@ExceptionHandler(DuplicateSpittleException.class)
	public String duplicateSpittleHandler() {
		return "error/duplicate";
	}
}

如果任何地方抛出DuplicateSpittleException,则会调用duplicateSpittleHandler方法来处理异常。

7.5 跨重定向请求传递数据

在5.4说了重定向的基础功能,Spring还为其提供了其他的辅助功能。

Q:正在发起重定向功能的方法该如何发送数据给重定向的目标方法呢?

当一个处理器方法完成之后,该方法所指定的模型数据将会复制到请求中,并作为请求中的属性,请求会转发到视图上进行渲染。因为控制器方法和视图所处理的是同一个请求,所以在转发中请求属性能够得以保存。

以上是请求的处理内容,但是重定向不同。
当控制器返回的结果是重定向的话,原始的请求就结束了,并且会发起新的一个GET请求。原始请求中带有的参数也会一起消失,在新请求属性中,没有任何来自模型的数据,必须通过方法内获取后,赋值到重定向的结果中,作为新请求的数据。

重定向传递数据的两种方法:

  • 使用URL模板以路径变量和/或查询参数的形式传递数据。
  • 通过flash属性发送数据。

第一种方法

	@RequestMapping(value="/register",method=POST)
	public String executeRegister(Spittr spittr,Model model) {\
		spitterRepository.save(spittr);
		model.addAttribute("username",spitter.getUsername());
		// 假设有一个POJO类为spittr,且有一个变量为Username,并生成一个getUsername()的方法
		// 5.4中的写法
		// return "redirect:/home/" + spittr.getUsername()
		return "redirect:/home/{username}";

	}

将username作为占位符填充到URL模板中,而不是直接连接字符串,在占位符中重新转义编码,提升了一定的安全性。

例2:

	@RequestMapping(value="/register",method=POST)
	public String executeRegister(Spittr spittr,Model model) {\
		spitterRepository.save(spittr);
		model.addAttribute("username",spitter.getUsername());
		model.addAttribute("userId",spitter.getId());
		// 假设有一个POJO类为spittr,且有一个变量为Username,并生成一个getUsername()的方法
		// 5.4中的写法
		// return "redirect:/home/" + spittr.getUsername()
		return "redirect:/home/{username}";
	}

新增userId,但是在重定向路径中并没有userId的占位符,重定向后它会被加载到尾部,假设ID=47,生成
/home/lihua?userId=47。有没有对这个url比较熟悉,?后面的内容就是这样子加上去的。

该方法的局限性:在于它只可以发送简单的值,比如字符串或者数字等,没办法处理更为复杂的值。处理复杂的值就是flash方法存在的意义。

第二章方法--flash

如果要发送实际的对象等,Spring会将胯重定向存活的数据放到会话中,提供将数据发送为flash属性的功能,按照定义,flash属性会一直携带这些数据直到下一次请求。提供RedirectAttributes设置flash属性的方法。该模块不仅包含了Model的所有功能,还提供几个方法设置flash属性。

	@RequestMapping(value="/register",method=POST)
	public String executeRegister(Spittr spittr,RedirectAttributes model) {\
		spitterRepository.save(spittr);
		model.addAttribute("username",spitter.getUsername());
		// 假设有一个POJO类为spittr,且有一个变量为Username,并生成一个getUsername()的方法
		// 5.4中的写法
		// return "redirect:/home/" + spittr.getUsername();
		model.addFlashAttribute("spitter",spitter);
		return "redirect:/home/{username}";

	}

"spitter"为key,后面spitter为值,而flash会自动进行推断而不需要设置。

posted @ 2020-05-31 16:02  鱼换水  阅读(...)  评论(...编辑  收藏