@ControllerAdvice 用法解析
@ControllerAdvice 用法
顧名思義,@ControllerAdvice就是@Controller 的增強(qiáng)版。@ControllerAdvice主要用來處理全局?jǐn)?shù)據(jù),一般搭配@ExceptionHandler、@ModelAttribute以及@InitBinder使用。
全局異常處理
@ControllerAdvice最常見的使用場(chǎng)景就是全局異常處理。比如文件上傳大小限制的配置,如果用戶上傳的文件超過了限制大小,就會(huì)拋出異常,此時(shí)可以通過@ControllerAdvice結(jié)合@ExceptionHandler定義全局異常捕獲機(jī)制,代碼如下:
@ControllerAdvice
public class CustomExceptionHandler {
@ExceptionHandler(MaxUploadSizeExceededException.class)
public void uploadException(MaxUploadSizeExceededException e, HttpServletResponse resp) throws IOException {
resp.setContentType("text/html;charset=utf-8");
System.out.println(1111);
PrintWriter out = resp.getWriter();
out.write("上傳文件大小超出限制!");
out.flush();
out.close();
}
}只需在系統(tǒng)中定義CustomExceptionHandler類,然后添加@ControllerAdvice注解即可。當(dāng)系統(tǒng)啟動(dòng)時(shí),該類就會(huì)被掃描到Spring容器中,然后定義uploadException方法,在該方法上添加了@ExceptionHandler注解,其中定義的MaxUploadSizeExceededException.class 表明該方法用來處理MaxUploadSizeExceededException類型的異常。如果想讓該方法處理所有類型的異常,只需將MaxUploadSizeExceededException改為 Exception即可。方法的參數(shù)可以有異常實(shí)例、HttpServletResponse以及HttpServletRequest、Model 等,返回值可以是一段JSON、一個(gè)ModelAndView、一個(gè)邏輯視圖名等。此時(shí),上傳一個(gè)超大文件會(huì)有錯(cuò)誤提示給用戶。

如果返回參數(shù)是一個(gè)ModelAndView,假設(shè)使用的頁面模板為Thymeleaf(注意添加Thymeleaf相關(guān)依賴),此時(shí)異常處理方法定義如下:
@ControllerAdvice
public class CustomExceptionHandler {
@ExceptionHandler(MaxUploadSizeExceededException.class)
public ModelAndView uploadException(MaxUploadSizeExceededException e) throws IOException {
ModelAndView mv = new ModelAndView();
mv.addObject("msg", "上傳文件大小超出限制! ");
mv.setViewName("error");
return mv;
}
}然后在resources/templates目錄下創(chuàng)建error.html文件,內(nèi)容如下:
<!DOCTYPE html>
<html lang="en" xmlns:th="http://www.thymeleaf.org">
<head>
<meta charset="UTF-8">
<title>Title</title></head>
<body>
<div th:text="${msg}"></div>
</body>
</html>此時(shí)上傳出錯(cuò)效果一致。
添加全局?jǐn)?shù)據(jù)
@ControllerAdvice是一個(gè)全局?jǐn)?shù)據(jù)處理組件,因此也可以在@ControllerAdvice中配置全局?jǐn)?shù)據(jù),使用@ModelAttribute注解進(jìn)行配置,代碼如下:
@ControllerAdvice
public class GlobalConfig {
@ModelAttribute(value = "info")
public Map<String, String> userInfo() {
HashMap<String, String> map = new HashMap<>();
map.put("username", "羅貫中");
map.put("gender", "男");
return map;
}
}代碼解釋:
- 在全局配置中添加userInfo方法,返回一個(gè)map。該方法有一個(gè)注解@ModelAttribute,其中的value屬性表示這條返回?cái)?shù)據(jù)的key,而方法的返回值是返回?cái)?shù)據(jù)的value。
- 此時(shí)在任意請(qǐng)求的Controller 中,通過方法參數(shù)中的Model都可以獲取info 的數(shù)據(jù)。
Controller 例代碼如下:
public class MyController {
@GetMapping("/hello")
@ResponseBody
public void hello(Model model) {
Map<String, Object> map = model.asMap();
Set<String> keySet = map.keySet();
Iterator<String> iterator = keySet.iterator();
while (iterator.hasNext()) {
String key = iterator.next();
Object value = map.get(key);
System.out.println(key + ">>>>>" + value);
}
}
}在請(qǐng)求方法中,將Model 中的數(shù)據(jù)打印出來,如圖所示。

請(qǐng)求參數(shù)預(yù)處理
@ControllerAdvice結(jié)合@InitBinder還能實(shí)現(xiàn)請(qǐng)求參數(shù)預(yù)處理,即將表單中的數(shù)據(jù)綁定到實(shí)體類上時(shí)進(jìn)行一些額外處理。
例如有兩個(gè)實(shí)體類 Book和 Author,代碼如下:
@Data
@AllArgsConstructor
@NoArgsConstructor
@Component
@ToString
public class Book {
private String name;
private String author;
@JsonIgnore//一般標(biāo)記在屬性或者方法上,返回的json數(shù)據(jù)即不包含該屬性
private Float price;
@JsonFormat(pattern = "yyyy-MM-dd")
private Date publicationDate;
}
@Data
@AllArgsConstructor
@NoArgsConstructor
@ToString
public class Author {
private String name;
private int age;
}在 Controller 上需要接收兩個(gè)實(shí)體類的數(shù)據(jù),Controller 中的方法定義如下:
@ControllerAdvice
public class GlobalConfig1 {
@InitBinder("b")
public void init(WebDataBinder binder) {
binder.setFieldDefaultPrefix("b.");
}
@InitBinder("a")
public void init2(WebDataBinder binder) {
binder.setFieldDefaultPrefix("a.");
}
}代碼解釋:
- 在 GlobalConfig類中創(chuàng)建兩個(gè)方法,第一個(gè)@InitBinder(“b”)表示該方法是處理@ModelAttribute(“b”)對(duì)應(yīng)的參數(shù)的,第二個(gè)@InitBinder(“a”)表示該方法是處理@ModelAttribute(“a”)對(duì)應(yīng)的參數(shù)的。
- 在每個(gè)方法中給相應(yīng)的 Field設(shè)置一個(gè)前綴,然后在瀏覽器中請(qǐng)求http:/ocalhost:8080/book?b.name=三國(guó)演義&b.author=羅貫中&a.name=曹雪芹&a.age=48,即可成功地區(qū)分出name屬性。
- 在WebDataBinder對(duì)象中,還可以設(shè)置允許的字段、禁止的字段、必填字段以及驗(yàn)證器等。
ld設(shè)置一個(gè)前綴,然后在瀏覽器中請(qǐng)求http:/ocalhost:8080/book?b.name=三國(guó)演義&b.author=羅貫中&a.name=曹雪芹&a.age=48,即可成功地區(qū)分出name屬性。
- 在WebDataBinder對(duì)象中,還可以設(shè)置允許的字段、禁止的字段、必填字段以及驗(yàn)證器等。

到此這篇關(guān)于@ControllerAdvice 用法的文章就介紹到這了,更多相關(guān)@ControllerAdvice 用法內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
- SpringBoot @ExceptionHandler與@ControllerAdvice異常處理詳解
- SpringBoot整合Web開發(fā)之文件上傳與@ControllerAdvice
- SpringMVC中@controllerAdvice注解的詳細(xì)解釋
- 基于@RestControllerAdvice與@ControllerAdvice的區(qū)別說明
- 使用@ControllerAdvice同時(shí)配置過濾多個(gè)包
- 解決spring @ControllerAdvice處理異常無法正確匹配自定義異常
- SpringBoot @ControllerAdvice 攔截異常并統(tǒng)一處理
相關(guān)文章
java.net.MalformedURLException異常的解決方法
下面小編就為大家?guī)硪黄猨ava.net.MalformedURLException異常的解決方法。小編覺得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧2017-05-05
SpringBoot集成Apache POI實(shí)現(xiàn)Excel的導(dǎo)入導(dǎo)出
Apache POI是一個(gè)流行的Java庫(kù),用于處理Microsoft Office格式文件,包括Excel文件,本文主要介紹了SpringBoot集成Apache POI實(shí)現(xiàn)Excel的導(dǎo)入導(dǎo)出,具有一定的參考價(jià)值,感興趣的可以了解一下2024-06-06
Java中實(shí)現(xiàn) SHA-256加密的兩種方式
這篇文章主要介紹了Java中實(shí)現(xiàn) SHA-256加密的兩種方式,本文通過實(shí)例代碼給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友參考下吧2024-01-01
Spring Boot 自定義 Shiro 過濾器無法使用 @Autowired問題及解決方法
這篇文章主要介紹了Spring Boot 自定義 Shiro 過濾器無法使用 @Autowired問題及解決方法 ,本文給大家介紹的非常詳細(xì),具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2019-06-06
SpringBoot配置文件方式,在線yml文件轉(zhuǎn)properties
這篇文章主要介紹了SpringBoot配置文件方式,在線yml文件轉(zhuǎn)properties,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2022-07-07
Spring?Security實(shí)現(xiàn)接口放通的方法詳解
在用Spring?Security項(xiàng)目開發(fā)中,有時(shí)候需要放通某一個(gè)接口時(shí),我們需要在配置中把接口地址配置上,這樣做有時(shí)候顯得麻煩。本文將通過一個(gè)注解的方式快速實(shí)現(xiàn)接口放通,感興趣的可以了解一下2022-05-05
Android應(yīng)用開發(fā)之將SQLite和APK一起打包的方法
這篇文章主要介紹了Android應(yīng)用開發(fā)之將SQLite和APK一起打包的方法,文章時(shí)間較早,盡管現(xiàn)在開發(fā)環(huán)境已大都遷移至Android Studio上,但打包原理依然相同,需要的朋友可以參考下2015-08-08

