1、在购物车进行结算时,发现页面报错
2、根据提示,发现是远程查询所有购物项信息( cartFeignService.getUserCartItems()) 时出问题
3、在方法中,虽然当前用户一定已经登录,但是 userInfoTo 获取到的是临时用户信息,无法根据 userId 获取 redis 键值,报错
4、发生错误的原因:
4-1、在远程查询所有购物项信息时点击 step into 单步执行
List<OrderItemVo> items = cartFeignService.getUserCartItems();
4-2、进入 ReflectiveFeign 类的 invoke() 方法,使用代理对象实际执行的方法
4-3、进入 SynchronousMethodHandler 的 invoke() 方法,首先创建了一个新的请求模板
4-4、随后调用 SynchronousMethodHandler 的 executeAndDecode() 方法对模板进行处理
4-5、在 SynchronousMethodHandler 的 targetRequest() 方法中,遍历所有请求拦截器的迭代器来包装新模板,实际调试中却发现:包装后的模板与新模板并没有什么区别
4-6、在先前的设计中,当前用户的登录信息被存放在cookie中,每次请求都会携带
4-7、cart 模块的拦截器中,就是通过 cookie 里存放的 sessionId 来判断用户是否登录的。现在请求头中不携带任何数据,无法获取登录用户信息,会被认为没有登录,导致后续的错误
5、解决措施:
5-1、在 order 模块创建一个拦截器,在新请求模板中添加 cookie
@Configuration
public class MyFeignConfig {
@Bean
public RequestInterceptor requestInterceptor(){
return new RequestInterceptor() {
@Override
public void apply(RequestTemplate requestTemplate) {
ServletRequestAttributes requestAttributes = (ServletRequestAttributes) RequestContextHolder.getRequestAttributes();
HttpServletRequest request = requestAttributes.getRequest();
String cookie = request.getHeader("Cookie");
requestTemplate.header("Cookie", cookie);
}
};
}
}
5-2、验证:模板中有请求头数据,请求中有 sessionId