问题
今天在写 Java如何获取 IP 属地
的博客时,使用谷歌浏览器调用后端获取ip完整地址
的接口时,出现了如下图这个错误:
Whitelabel Error Page
This application has no explicit mapping for /error, so you are seeing this as a fallback.
Mon Aug 01 15:36:11 CST 2022
There was an unexpected error (type=Internal Server Error, status=500).
status = 500
说明是服务器端出现了,服务器端怎么会出现错误呢?于是如下分析
分析问题
获取ip地址的源代码
package com.example.demo;
import lombok.extern.slf4j.Slf4j;
import org.springframework.http.HttpHeaders;
import org.springframework.http.server.ServerHttpRequest;
import java.net.InetAddress;
import java.net.UnknownHostException;
/**
* @author zs
* @datetime 2022/8/1 10:30
* @desc 查询IP地址
*/
@Slf4j
public class IpAddress {
/**
* @author zs
* @datetime 2022/8/1:14:39
* @desc 获取ip地址
*/
public static String getIpAddress(ServerHttpRequest request) {
HttpHeaders headers = request.getHeaders();
String ipAddress = headers.getFirst("X-Forwarded-For");
if (ipAddress == null || ipAddress.length() == 0 || "unknown".equalsIgnoreCase(ipAddress)) {
ipAddress = headers.getFirst("Proxy-Client-IP");
}
if (ipAddress == null || ipAddress.length() == 0 || "unknown".equalsIgnoreCase(ipAddress)) {
ipAddress = headers.getFirst("WL-Proxy-Client-IP");
}
if (ipAddress == null || ipAddress.length() == 0 || "unknown".equalsIgnoreCase(ipAddress)) {
ipAddress = request.getRemoteAddress().getAddress().getHostAddress();
if (ipAddress.equals("127.0.0.1") || ipAddress.equals("0:0:0:0:0:0:0:1")) {
// 根据网卡取本机配置的IP
try {
InetAddress inet = InetAddress.getLocalHost();
ipAddress = inet.getHostAddress();
} catch (UnknownHostException e) {
log.error("根据网卡获取本机配置的IP异常", e);
}
}
}
// 对于通过多个代理的情况,第一个IP为客户端真实IP,多个IP按照','分割
if (ipAddress != null && ipAddress.indexOf(",") > 0) {
ipAddress = ipAddress.split(",")[0];
}
return ipAddress;
}
}
接口调用源代码
package com.example.demo.controller;
import org.springframework.http.server.ServerHttpRequest;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;
import static com.example.demo.IpAddress.getIpAddress;
/**
* @author zs
* @datetime 2022/7/22 10:14
* @desc
*/
@RestController
@RequestMapping("/test")
public class IndexController {
@GetMapping("/getIpAddress")
public Object get(ServerHttpRequest request) {
String ipAddress = getIpAddress(request);
return ipAddress;
}
}
后端错误
后端报出的错误如下图所示:
java.lang.IllegalStateException: No primary or single unique constructor found for interface org.springframework.http.server.ServerHttpRequest
从图中可以看到,无法实例化ServerHttpRequest
的构造器,原因如下:
-
用WebFlux时,获取请求头的信息时使用:ServerHttpRequest
-
用MVC时(或WebFlux、MVC依赖同时存在时),获取http请求头的信息使用:HttpServletRequest
而我没有使用WebFlux
,因而,我不能使用ServerHttpRequest
来接参,所以,应该使用HttpServletRequest
来接参。
解决问题
修改获取ip地址的源代码
package com.example.demo;
import lombok.extern.slf4j.Slf4j;
import javax.servlet.http.HttpServletRequest;
import java.net.InetAddress;
import java.net.UnknownHostException;
/**
* @author zs
* @datetime 2022/8/1 10:30
* @desc 查询IP地址
*/
@Slf4j
public class IpAddress {
/**
* @author zs
* @datetime 2022/8/1:14:39
* @desc 获取ip地址
*/
public static String getIpAddress(HttpServletRequest request) {
String ipAddress = request.getHeader("X-Forwarded-For");
if (ipAddress == null || ipAddress.length() == 0 || "unknown".equalsIgnoreCase(ipAddress)) {
ipAddress = request.getHeader("Proxy-Client-IP");
}
if (ipAddress == null || ipAddress.length() == 0 || "unknown".equalsIgnoreCase(ipAddress)) {
ipAddress = request.getHeader("WL-Proxy-Client-IP");
}
if (ipAddress == null || ipAddress.length() == 0 || "unknown".equalsIgnoreCase(ipAddress)) {
ipAddress = request.getRemoteAddr();
if (ipAddress.equals("127.0.0.1") || ipAddress.equals("0:0:0:0:0:0:0:1")) {
// 根据网卡取本机配置的IP
try {
InetAddress inet = InetAddress.getLocalHost();
ipAddress = inet.getHostAddress();
} catch (UnknownHostException e) {
log.error("根据网卡获取本机配置的IP异常", e);
}
}
}
// 对于通过多个代理的情况,第一个IP为客户端真实IP,多个IP按照','分割
if (ipAddress != null && ipAddress.indexOf(",") > 0) {
ipAddress = ipAddress.split(",")[0];
}
return ipAddress;
}
}
接口调用源代码
package com.example.demo.controller;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;
import javax.servlet.http.HttpServletRequest;
import static com.example.demo.IpAddress.getIpAddress;
/**
* @author zs
* @datetime 2022/7/22 10:14
* @desc
*/
@RestController
@RequestMapping("/test")
public class IndexController {
@GetMapping("/getIpAddress")
public Object get(HttpServletRequest request) {
String ipAddress = getIpAddress(request);
return ipAddress;
}
}
从新启动项目,即可获取ip地址
了,如下图所示:
汇总status=404可能出现的错误原因以及解决方案
出现这个异常,说明跳转页面的url无对应的值。
如果以上解决方案无法解决你的问题,可以参考一下解决方案。
解决方案及原因1
Application启动类
的位置不对,要将Application类
放在最外侧,即包含所有子包 。
原因:spring-boot
会自动加载启动类所在包下及其子包下的所有组件.
解决方案及原因2
在springboot的配置文件:application.yml或application.properties
中关于视图解析器的配置问题:
-
当pom文件下的spring-boot-starter-parent版本高时使用:
spring.mvc.view.prefix/spring.mvc.view.suffix
-
当pom文件下的spring-boot-starter-parent版本低时使用:
spring.view.prefix/spring.view.suffix
解决方案及原因3
控制器的URL路径书写问题 :@RequestMapping(“xxxxxxxxxxxxxx”)
实际访问的路径与”xxx”不符合。