使用Spring boot怎么自定義http的反饋狀態(tài)碼?針對(duì)這個(gè)問(wèn)題,這篇文章詳細(xì)介紹了相對(duì)應(yīng)的分析和解答,希望可以幫助更多想解決這個(gè)問(wèn)題的小伙伴找到更簡(jiǎn)單易行的方法。
創(chuàng)新互聯(lián)于2013年成立,先為果洛州等服務(wù)建站,果洛州等地企業(yè),進(jìn)行企業(yè)商務(wù)咨詢服務(wù)。為果洛州企業(yè)網(wǎng)站制作PC+手機(jī)+微官網(wǎng)三網(wǎng)同步一站式服務(wù)解決您的所有建站問(wèn)題。
通過(guò)spring boot構(gòu)建一些web程序,這些web程序之間通過(guò)http進(jìn)行數(shù)據(jù)訪問(wèn)、共享,如下圖,
假設(shè)現(xiàn)在client發(fā)起一次保存數(shù)據(jù)的請(qǐng)求到server,server可能會(huì)返回如下類似的數(shù)據(jù)
{ "status":1, "message":"xxxxxx" }
然后client通過(guò)解析json獲得status來(lái)判斷當(dāng)前的請(qǐng)求操作是否成功,開(kāi)發(fā)過(guò)程中通過(guò)都是這么做的,但是這樣在restful設(shè)計(jì)中不怎么好,其實(shí)這個(gè)status字段的表達(dá)完全可以通過(guò)http status
來(lái)表示,類似404、500、502這種都有明確的定義并且相互理解、溝通起來(lái)也方便。
文章主要記錄一下我是如何在spring boot中實(shí)現(xiàn)自定反饋狀態(tài)碼的,以及我找到的三種實(shí)現(xiàn)方式。
第一種,使用**@ResponseStatus** 。這是一個(gè)注解,可以作用在方法和類上面,如下使用,
在方法上使用方式,
@RequestMapping(value = "/user", method = RequestMethod.GET) @ResponseStatus(code=HttpStatus.INTERNAL_SERVER_ERROR,reason="server error") public String getUser(){ return "im zhangsan"; }
啟動(dòng)web程序,通過(guò)postman訪問(wèn)http://127.0.0.1:8100/user,會(huì)出現(xiàn)下面結(jié)果,
{ "timestamp": 1497850427325, "status": 500, "error": "Internal Server Error", "message": "server error", "path": "/user" }
這里我一開(kāi)始覺(jué)得很奇怪,為什么我的getUser方法中沒(méi)有錯(cuò)誤,結(jié)果還是出現(xiàn)了500錯(cuò)誤?原因就是@ResponseStatus
注解的問(wèn)題,我后面猜測(cè)它會(huì)強(qiáng)制的將映射轉(zhuǎn)化成500的狀態(tài)碼。這種應(yīng)用場(chǎng)景我想不太明白在什么地方會(huì)用到。
在類中使用方式,
@ResponseStatus(code=HttpStatus.INTERNAL_SERVER_ERROR,reason="111") public class ServerException extends Exception { }
這種使用方式就是將自定義異常和狀態(tài)碼結(jié)合在一起,合理使用自定義異常機(jī)制可以最大化的提高程序的健壯性,下面看如何使用,
@RequestMapping(value = "/user", method = RequestMethod.GET) public String getUser(@RequestParam String userName) throws ServerException{ if(StringUtils.isEmpty(userName)){ throw new ServerException(); } return "im zhangsan"; }
這段代碼的意思是當(dāng)userName字段為null的時(shí)候會(huì)拋出ServerException異常,但是ServerException類被標(biāo)記了@ResponseStatus
注解,因此會(huì)直接報(bào)500錯(cuò)誤,如果覺(jué)得500不適合還可以定義其它的錯(cuò)誤代碼。
這種方式看著已經(jīng)很好了,可以按照邏輯自定義反饋碼,程序夠健壯。這種方式也有不好地方,如果反饋碼太多需要定義太多的異常類,并且錯(cuò)誤內(nèi)容reason還是不能手動(dòng)定義。
到這里,我基本上放棄了@ResponseStatus
的使用了。
第二種,使用HttpServletResponse,HttpServletResponse是javax.servlet下的一個(gè)接口,如下使用,
@RequestMapping(value = "/user", method = RequestMethod.GET) public void getUser(HttpServletResponse response) throws IOException{ response.setStatus(500); response.getWriter().append("server error"); }
這種方式可以很好的實(shí)現(xiàn)同時(shí)滿足自定義反饋碼+消息內(nèi)容,一般的實(shí)現(xiàn)方式也都是這樣。但是這樣也不是太好,
response.setContentType("application/json");
和response.setCharacterEncoding("UTF-8");
,這樣做有些多余,重復(fù)的工作太多,雖然可以進(jìn)行封裝。@ResponseBody
出現(xiàn)沖突,其次就是不能利用@ResponseBody
自動(dòng)封裝json的特性,在spring mvc框架中如果在方法上加上@ResponseBody
是可以對(duì)返回值自動(dòng)進(jìn)行json封裝的。再找找其他的,如果沒(méi)有找到,估計(jì)也只能接受這個(gè)不完美的東西了。
后來(lái)在翻閱spring boot文檔的時(shí)候找到了ResponseEntity這么一個(gè)東西,這就是我要說(shuō)的第三種方式。
第三種,使用ResponseEntity
不多說(shuō),直接上代碼,
@RequestMapping(value = "/user", method = RequestMethod.GET) public ResponseEntity
通過(guò)postman查看返回結(jié)果,如下,
{ "name": "zhangsan" }
可以直接將map對(duì)象幫我轉(zhuǎn)化成json對(duì)象,并且可以獲得自定義狀態(tài)碼,很好,很強(qiáng)大。
這種方式很和我意,
相比于前面兩種,這種方式很對(duì)我胃口。
仔細(xì)看了ResponseEntity的說(shuō)明,發(fā)現(xiàn)spring mvc其它很多地方也都有使用,如下,下面內(nèi)容摘自org.springframework.http.ResponseEntity
文件注釋,
In RestTemplate, this class is returned by getForEntity() and exchange() :
ResponseEntityentity = template.getForEntity("http://example.com", String.class); String body = entity.getBody(); MediaType contentType = entity.getHeaders().getContentType(); HttpStatus statusCode = entity.getStatusCode();
Can also be used in Spring MVC, as the return value from a @Controller method:
@RequestMapping("/handle") public ResponseEntityhandle() { URI location = ...; HttpHeaders responseHeaders = new HttpHeaders(); responseHeaders.setLocation(location); responseHeaders.set("MyResponseHeader", "MyValue"); return new ResponseEntity ("Hello World", responseHeaders, HttpStatus.CREATED); }
這就是上面說(shuō)過(guò)的。
Or, by using a builder accessible via static methods:
@RequestMapping("/handle") public ResponseEntityhandle() { URI location = ...; return ResponseEntity.created(location).header("MyResponseHeader", "MyValue").body("Hello World"); }
自定義http反饋碼在設(shè)計(jì)優(yōu)良的restful api中起到關(guān)鍵作用,http反饋碼是業(yè)內(nèi)統(tǒng)一、共識(shí)的,建議在盡量不要通過(guò)解析json來(lái)獲得status判斷操作結(jié)果。
關(guān)于使用Spring boot怎么自定義http的反饋狀態(tài)碼問(wèn)題的解答就分享到這里了,希望以上內(nèi)容可以對(duì)大家有一定的幫助,如果你還有很多疑惑沒(méi)有解開(kāi),可以關(guān)注創(chuàng)新互聯(lián)行業(yè)資訊頻道了解更多相關(guān)知識(shí)。