1. 引言

本教程中,将展示如何从REST Controller中获取HTTP 请求头。
我将使用@RequestHeader注解分别来获取单个Header和所有Header, 然后将深入了解下@RequestHeader的属性。

2. 获取Header

2.1. 获取单个Header

如果我们要获取具体的Header,可以在@RequestHeader指定对应的Header名称。

  1. @GetMapping("/greeting")
  2. public ResponseEntity<String> greeting(@RequestHeader("accept-language") String language) {
  3. // code that uses the language variable
  4. return new ResponseEntity<String>(greeting, HttpStatus.OK);
  5. }

然后我们可以在方法中使用该变量的名称来获取请求头的值,以上代码中,如果accept-language无法在请求头中找到,则会返回400 Bad Request错误。
当然,请求头不一定非要是字符串。例如,如果知道请求头的值是数字,可以将变量声明为数字类型。

  1. @GetMapping("/double")
  2. public ResponseEntity<String> doubleNumber(@RequestHeader("my-number") int myNumber) {
  3. return new ResponseEntity<String>(String.format("%d * 2 = %d",
  4. myNumber, (myNumber * 2)), HttpStatus.OK);
  5. }

2.2. 一次获取所有Header

如果不确定哪些Header会被使用,或者想要使用某些未在方法中声明的header, 那么在使用@RequestHeader注解时,不指定名称即可。
可以使用诸如Map, MultiValueMap,或者HttpHeader对象来接收变量,有很多种选择。 先看下使用Map来接收。

  1. @GetMapping("/listHeaders")
  2. public ResponseEntity<String> listAllHeaders(@RequestHeader Map<String, String> headers) {
  3. headers.forEach((key, value) -> {
  4. LOG.info(String.format("Header '%s' = %s", key, value));
  5. });
  6. return new ResponseEntity<String>(String.format("Listed %d headers", headers.size()), HttpStatus.OK);
  7. }

如果使用Map,则当某个Header有多个值时,只能拿到第一个值。这等价于我们使用MultiValueMap时,调用getFirst方法。
如果Header中有多个参数,可以使用MultiValueMap来接收参数值。

  1. @GetMapping("/multiValue")
  2. public ResponseEntity<String> multiValue(@RequestHeader MultiValueMap<String, String> headers) {
  3. headers.forEach((key, value) -> {
  4. LOG.info(String.format("Header '%s' = %s", key, value.stream().collect(Collectors.joining("|"))));
  5. });
  6. return new ResponseEntity<String>(String.format("Listed %d headers", headers.size()), HttpStatus.OK);
  7. }

也可以使用HttpHeader对象:

  1. @GetMapping("/getBaseUrl")
  2. public ResponseEntity<String> getBaseUrl(@RequestHeader HttpHeaders headers) {
  3. InetSocketAddress host = headers.getHost();
  4. String url = "http://" + host.getHostName() + ":" + host.getPort();
  5. return new ResponseEntity<String>(String.format("Base URL = %s", url), HttpStatus.OK);
  6. }

HttpHeader对象中可以访问应用的一些公共请求头。
使用以上三种方式来获取请求头时,如果Header不存在,将会返回null值。

3. @RequestHeader的属性

我们已经学习了使用@RequestHeader注解访问请求头的基础知识,现在来进一步了解它的属性。
我们已经隐式地使用了name或值value,当我们指定我们的头:

  1. public ResponseEntity<String> greeting(@RequestHeader("accept-language") String language) {
  2. }

当然也可以用name属性:

  1. public ResponseEntity<String> greeting(@RequestHeader(name = "accept-language") String language) {
  2. }

同样地可以使用value属性:

  1. public ResponseEntity<String> greeting(@RequestHeader(value = "accept-language") String language) {
  2. }

但我们明确地指定某个Header的名称时,默认这个Header是必需的,如果请求中不存在,将会发生400错误。
可以使用required属性来表明某个Header不是必需的。

  1. @GetMapping("/nonRequiredHeader")
  2. public ResponseEntity<String> evaluateNonRequiredHeader(
  3. @RequestHeader(value = "optional-header", required = false) String optionalHeader) {
  4. return new ResponseEntity<String>(
  5. String.format("Was the optional header present? %s!", (optionalHeader == null ? "No" : "Yes")),
  6. HttpStatus.OK);
  7. }

当某个Header不存在时,我们声明的变量可能为null值,因此,程序中要适当对null值进行检查。
或者使用defaultValue提供默认的header值。

  1. @GetMapping("/default")
  2. public ResponseEntity<String> evaluateDefaultHeaderValue(
  3. @RequestHeader(value = "optional-header", defaultValue = "3600") int optionalHeader) {
  4. return new ResponseEntity<String>(String.format("Optional Header is %d", optionalHeader),
  5. HttpStatus.OK);
  6. }

4. 总结

在这篇简短教程中,我们学习了如何在Spring REST控制器中获取请求头。首先,我们使用@RequestHeader注解向控制器方法提供请求头。
在了解了基础知识之后,我们详细查看了@RequestHeader注解释的一些属性。
示例代码可以在GitHub上找到。