以SpringCloud构建的微服务系统为例,使用前后端分离的架构,每个系统都会提供一些通用的请求参数,例如移动端的系统版本信息、IMEI信息,Web端的IP信息,浏览器版本信息等,这些参数可能放在header里,也可以放在参数里,如果这些参数需要在每个方法内声明定义,一来工作量太大,二是这些通用参数与业务接口方法耦合过紧,本身就是一个不好的设计。

这个问题该如何优雅地解决呢?

  • 利用SpringMVC提供拦截器,对匹配的请求,抽取通用的header信息(假设通用字段全部放在header里)
  • 将每个请求的信息单独隔离开,互不干扰。
  • Controller层使用时,可以将在该请求线程(http线程)里将通用的header信息提取出来使用。
  • 请求线程完成时,相应的header头信息对象需要回收销毁。
  • SpringMVA提供的HandlerInterceptorAdapter可以拿来使用,继承实现即可。
  • 使用ThreadLocal记录每个请求的信息,ThreadLocal有隔离线程变量的作用。
  1. public abstract class HandlerInterceptorAdapter implements AsyncHandlerInterceptor {
  2. @Override
  3. public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler)
  4. throws Exception {
  5. // 在业务接口方法处理之前被调用,可以在这里对通用的header信息进行提取
  6. return true;
  7. }
  8. @Override
  9. public void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler,
  10. @Nullable ModelAndView modelAndView) throws Exception {
  11. // 这个方法在业务接口方法执行完成后,生成SpringMVC ModelAndView之前被调用
  12. // 今天这个案例我们不用此方法,故可以不实现。
  13. }
  14. @Override
  15. public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler,
  16. @Nullable Exception ex) throws Exception {
  17. // 这个方法在DispatcherServlet完全处理完成后被调用,可以在这里对ThreadLocal的内容进行释放
  18. }
  19. @Override
  20. public void afterConcurrentHandlingStarted(HttpServletRequest request, HttpServletResponse response,
  21. Object handler) throws Exception {
  22. // 这个方法用来处理异步主动,但也会先行调用preHandle,然后执行此方法,异步线程完成后会执行postHandle和afterCompletion两方法,这里暂时用不上。
  23. }
  24. }
  1. public class ThreadLocal<T> {
  2. protected T initialValue() {
  3. return null;
  4. }
  5. public T get() {
  6. // 获取当前的线程
  7. Thread t = Thread.currentThread();
  8. ThreadLocalMap map = getMap(t);
  9. if (map != null) {
  10. ThreadLocalMap.Entry e = map.getEntry(this);
  11. if (e != null) {
  12. @SuppressWarnings("unchecked")
  13. T result = (T)e.value;
  14. return result;
  15. }
  16. }
  17. return setInitialValue();
  18. }
  19. private T setInitialValue() {
  20. T value = initialValue();
  21. Thread t = Thread.currentThread();
  22. ThreadLocalMap map = getMap(t);
  23. if (map != null)
  24. map.set(this, value);
  25. else
  26. createMap(t, value);
  27. return value;
  28. }
  29. public void set(T value) {
  30. // 获取当前的线程
  31. Thread t = Thread.currentThread();
  32. ThreadLocalMap map = getMap(t);
  33. if (map != null)
  34. map.set(this, value);
  35. else
  36. createMap(t, value);
  37. }
  38. public void remove() {
  39. ThreadLocalMap m = getMap(Thread.currentThread());
  40. if (m != null)
  41. m.remove(this);
  42. }
  43. ThreadLocalMap getMap(Thread t) {
  44. return t.threadLocals;
  45. }
  46. void createMap(Thread t, T firstValue) {
  47. t.threadLocals = new ThreadLocalMap(this, firstValue);
  48. }
  49. }

简单来说,ThreadLocal最关键的get()和set()方法,都是针对当前线程来操作的,调用set()方法时把值放到ThreadMap(Map的一种实现)中,以当前线程的hash值为key,get()方法则对应以当前线程作为key来取值,从而实现每个线程的数据是隔离的效果。

另附上ThreadLocal类源码解读的导图,仅供参考

我们对实际业务系统进行简化处理,假定header信息固定有ip,uid,deviceId三个信息,按照上文的实现思路,开始案例演示。

通用的header信息,使用Dto对象进行封装:

  1. @Data
  2. public class CommonHeader implements Serializable {
  3. private static final long serialVersionUID = -3949488282201167943L;
  4. /**
  5. * 真实ip
  6. */
  7. private String ip;
  8. /**
  9. * 设备id
  10. */
  11. private String deviceId;
  12. /**
  13. * 用户uid
  14. */
  15. private Long uid;
  16. // 省略getter/setter/构造器
  17. }

定义Request请求的封装类Dto,并引入ThreadLocal:

  1. /**
  2. * 将公共请求头信息放在ThreadLocal中去
  3. */
  4. public class RequestWrap {
  5. private static ThreadLocal<CommonHeader> current = new ThreadLocal<>();
  6. /**
  7. * 获取静态的ThreadLocal对象
  8. * @return
  9. */
  10. public static ThreadLocal<CommonHeader> getCurrent() {
  11. return current;
  12. }
  13. /**
  14. * 获取ip
  15. * @return
  16. */
  17. public static String getIp() {
  18. CommonHeader request = current.get();
  19. if (request == null) {
  20. return StringUtils.EMPTY;
  21. }
  22. return request.getIp();
  23. }
  24. /**
  25. * 获取uid
  26. * @return
  27. */
  28. public static Long getUid() {
  29. CommonHeader request = current.get();
  30. if (request == null) {
  31. return null;
  32. }
  33. return request.getUid();
  34. }
  35. /**
  36. * 获取封装对象
  37. * @return
  38. */
  39. public static CommonHeader getCommonReq() {
  40. CommonHeader request = current.get();
  41. if (request == null) {
  42. return new CommonHeader(StringUtils.EMPTY, StringUtils.EMPTY,0L);
  43. }
  44. return request;
  45. }
  46. }

这里添加一个简单的工具类,将HttpServletRequest通过getHeader方法,生成CommonHeader类:

  1. public class HttpUtil {
  2. /**
  3. * 获取请求头信息
  4. *
  5. * @param request
  6. * @return
  7. */
  8. public static CommonHeader getCommonHeader(HttpServletRequest request) {
  9. String UID = request.getHeader("uid");
  10. Long uid = null;
  11. if (StringUtils.isNotBlank(UID)) {
  12. uid = Long.parseLong(UID);
  13. }
  14. return new CommonHeader(HttpUtil.getIp(request), request.getHeader("deviceId"), uid);
  15. }
  16. /**
  17. * 获取IP
  18. *
  19. * @param request
  20. * @return
  21. */
  22. public static String getIp(HttpServletRequest request) {
  23. String ip = request.getHeader("X-Forwarded-For");
  24. if (null != ip && !"".equals(ip.trim()) && !"unknown".equalsIgnoreCase(ip)) {
  25. int index = ip.indexOf(',');
  26. if (index != -1) {
  27. return ip.substring(0, index);
  28. } else {
  29. return ip;
  30. }
  31. }
  32. ip = request.getHeader("X-Real-IP");
  33. if (null != ip && !"".equals(ip.trim()) && !"unknown".equalsIgnoreCase(ip)) {
  34. return ip;
  35. }
  36. return request.getRemoteAddr();
  37. }
  38. }

最核心的实现终于出场了,这里继承HandlerInterceptorAdapter,这里作了简化处理:

  1. /**
  2. * 请求头处理
  3. *
  4. * @author yangfei
  5. */
  6. @Component
  7. public class BaseInterceptor extends HandlerInterceptorAdapter {
  8. private static final org.slf4j.Logger log = org.slf4j.LoggerFactory.getLogger(BaseInterceptor.class);
  9. @Override
  10. public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler)
  11. throws Exception {
  12. RequestWrap.getThreadLocal().set(HttpUtil.getCommonHeader(request));
  13. return true;
  14. }
  15. @Override
  16. public void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler,
  17. ModelAndView modelAndView) throws Exception {
  18. }
  19. @Override
  20. public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex)
  21. throws Exception {
  22. RequestWrap.getThreadLocal().remove();
  23. }
  24. @Override
  25. public void afterConcurrentHandlingStarted(HttpServletRequest request, HttpServletResponse response, Object handler)
  26. throws Exception {
  27. }
  28. }

如上一章节描述的逻辑,在preHandle方法内将request中的ip,uid,deviceId封装到RequestWrap对象里,在afterCompletion中对该线程的ThreadLocal值进行释放。

在Controller类的接口方法中,如要获取uid信息,只需要调用RequestWrap.getUid()方法即可,再也不需要在每个接口上声明uid参数了,如下示例:

  1. /**
  2. * 获取用户基础信息
  3. */
  4. @PostMapping(value = "/user/info")
  5. public Response<UserInfo> getUserInfo() {
  6. return userManager.getUserInfo(RequestWrap.getUid());
  7. }

这个实战的目标是解决通用header信息的在接口的重复定义问题,基于HandlerInterceptorAdapter拦截器的实现,ThreadLocal对线程访问数据的隔离来实现的,在实际生产项目应用中有很好的借鉴意义,希望对你有帮助。

专注Java高并发、分布式架构,更多技术干货分享与心得,请关注公众号:Java架构社区
可以扫左边二维码添加好友,邀请你加入Java架构社区微信群共同探讨技术
Java架构社区

版权声明:本文为huangying2124原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。
本文链接:https://www.cnblogs.com/huangying2124/p/13264753.html