背景

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

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

最佳实践

实现思路

  • 利用SpringMVC提供拦截器,对匹配的请求,抽取通用的header信息(假设通用字段全部放在header里)
  • 将每个请求的信息单独隔离开,互不干扰。
  • Controller层使用时,可以将在该请求线程(http线程)里将通用的header信息提取出来使用。
  • 请求线程完成时,相应的header头信息对象需要回收销毁。

实现方式

  • SpringMVA提供的HandlerInterceptorAdapter可以拿来使用,继承实现即可。
  • 使用ThreadLocal记录每个请求的信息,ThreadLocal有隔离线程变量的作用。

HandlerInterceptorAdapter的源码实现及注释

public abstract class HandlerInterceptorAdapter implements AsyncHandlerInterceptor {@Overridepublic boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler)throws Exception {// 在业务接口方法处理之前被调用,可以在这里对通用的header信息进行提取return true;}@Overridepublic void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler,@Nullable ModelAndView modelAndView) throws Exception {// 这个方法在业务接口方法执行完成后,生成SpringMVC ModelAndView之前被调用// 今天这个案例我们不用此方法,故可以不实现。}@Overridepublic void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler,@Nullable Exception ex) throws Exception {// 这个方法在DispatcherServlet完全处理完成后被调用,可以在这里对ThreadLocal的内容进行释放}@Overridepublic void afterConcurrentHandlingStarted(HttpServletRequest request, HttpServletResponse response,Object handler) throws Exception {// 这个方法用来处理异步主动,但也会先行调用preHandle,然后执行此方法,异步线程完成后会执行postHandle和afterCompletion两方法,这里暂时用不上。}
}

ThreadLocal的源码主要实现及注释

public class ThreadLocal<T> {protected T initialValue() {return null;}public T get() {// 获取当前的线程Thread t = Thread.currentThread();ThreadLocalMap map = getMap(t);if (map != null) {ThreadLocalMap.Entry e = map.getEntry(this);if (e != null) {@SuppressWarnings("unchecked")T result = (T)e.value;return result;}}return setInitialValue();}private T setInitialValue() {T value = initialValue();Thread t = Thread.currentThread();ThreadLocalMap map = getMap(t);if (map != null)map.set(this, value);elsecreateMap(t, value);return value;}public void set(T value) {// 获取当前的线程Thread t = Thread.currentThread();ThreadLocalMap map = getMap(t);if (map != null)map.set(this, value);elsecreateMap(t, value);}public void remove() {ThreadLocalMap m = getMap(Thread.currentThread());if (m != null)m.remove(this);}ThreadLocalMap getMap(Thread t) {return t.threadLocals;}void createMap(Thread t, T firstValue) {t.threadLocals = new ThreadLocalMap(this, firstValue);}
}

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

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

案例实战

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

DTO定义

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

@Data
public class CommonHeader implements Serializable {private static final long serialVersionUID = -3949488282201167943L;/*** 真实ip*/private String ip;/*** 设备id*/private String deviceId;/*** 用户uid*/private Long uid;// 省略getter/setter/构造器
}

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

/*** 将公共请求头信息放在ThreadLocal中去*/
public class RequestWrap {private static ThreadLocal<CommonHeader> current = new ThreadLocal<>();/*** 获取静态的ThreadLocal对象* @return*/public static ThreadLocal<CommonHeader> getCurrent() {return current;}/*** 获取ip* @return*/public static String getIp() {CommonHeader request = current.get();if (request == null) {return StringUtils.EMPTY;}return request.getIp();}/*** 获取uid* @return*/public static Long getUid() {CommonHeader request = current.get();if (request == null) {return null;}return request.getUid();}/*** 获取封装对象* @return*/public static CommonHeader getCommonReq() {CommonHeader request = current.get();if (request == null) {return new CommonHeader(StringUtils.EMPTY, StringUtils.EMPTY,0L);}return request;}
}

工具类

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

public class HttpUtil {/*** 获取请求头信息** @param request* @return*/public static CommonHeader getCommonHeader(HttpServletRequest request) {String UID = request.getHeader("uid");Long uid = null;if (StringUtils.isNotBlank(UID)) {uid = Long.parseLong(UID);}return new CommonHeader(HttpUtil.getIp(request), request.getHeader("deviceId"), uid);}/*** 获取IP** @param request* @return*/public static String getIp(HttpServletRequest request) {String ip = request.getHeader("X-Forwarded-For");if (null != ip && !"".equals(ip.trim()) && !"unknown".equalsIgnoreCase(ip)) {int index = ip.indexOf(',');if (index != -1) {return ip.substring(0, index);} else {return ip;}}ip = request.getHeader("X-Real-IP");if (null != ip && !"".equals(ip.trim()) && !"unknown".equalsIgnoreCase(ip)) {return ip;}return request.getRemoteAddr();}
}

拦截器类实现

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

/*** 请求头处理** @author yangfei*/
@Component
public class BaseInterceptor extends HandlerInterceptorAdapter {private static final org.slf4j.Logger log = org.slf4j.LoggerFactory.getLogger(BaseInterceptor.class);@Overridepublic boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler)throws Exception {RequestWrap.getThreadLocal().set(HttpUtil.getCommonHeader(request));return true;}@Overridepublic void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler,ModelAndView modelAndView) throws Exception {}@Overridepublic void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex)throws Exception {RequestWrap.getThreadLocal().remove();}@Overridepublic void afterConcurrentHandlingStarted(HttpServletRequest request, HttpServletResponse response, Object handler)throws Exception {}
}

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

业务接口方法的使用

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

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

总结

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

专注Java高并发、分布式架构,更多技术干货分享与心得,请关注公众号:Java架构社区

请求拦截_实战SpringCloud通用请求字段拦截处理相关推荐

  1. 请在请求中携带deviceid参数_实战SpringCloud通用请求字段拦截处理

    背景 以SpringCloud构建的微服务系统为例,使用前后端分离的架构,每个系统都会提供一些通用的请求参数,例如移动端的系统版本信息.IMEI信息,Web端的IP信息,浏览器版本信息等,这些参数可能 ...

  2. 5 拦截器拦截请求路由_手写简易版axios拦截器,实现微信小程序wx.request的封装与拦截...

    前言: axios是一个功能强大的网络请求库,其中拦截器又是axios的精髓.在小程序的开发或者需要手动实现ajax的时候,没有实现对请求的拦截,开发的时候非常不方便,因此手写一个简易版的axios拦 ...

  3. wx.chooseimage 超过了最大请求长度_一次 HTTP 请求到底经历了什么?

    作者:木木匠 链接:https://url.cn/5ER9kt2 今天这篇文章我们用抓包分析工具来分析 HTTP 请求是怎么样的? 环境准备 本来是想找个网站进行抓包分析的,但是正式环境的网站 HTT ...

  4. 怎么看java请求耗时_使用Filter计算请求耗时

    使用Filter计算请求耗时 有时为了更详细的检测web系统性能,需要计算每次请求到响应所耗费的时间,然后看看哪些请求耗时较多,从而有针对性的进行优化操作,此时可以使用Filter过滤器自己实现一个请 ...

  5. cefsharp 发送请求服务器_超高性能管线式HTTP请求(实践·原理·实现)

    来源:https://www.cnblogs.com/lulianqi/p/8167526.html 这里的高性能指的就是网卡有多快请求发送就能有多快,基本上一般的服务器在一台客户端的压力下就会出现明 ...

  6. 排队器拦截_过滤器(Filter)和拦截器(Interceptor)的执行顺序和区别

    首先建立一个SpringBoot工程,增加一个IndexController供测试使用. @Controller public class IndexController { @GetMapping( ...

  7. dueros模拟测试没有请求后台_实战 | 用手写一个骚气的请求合并,演绎底层的真实...

    来源:公众号[ java进阶架构师] 好文推荐: 字节跳动Java岗4面面经分享:索弓|+rabbitmq+spring+Redis 拼多多面经Java开发3面面经:准备好久没想到面试题超级简单 网易 ...

  8. http请求头获取请求链接_我们如何设计文件请求链接

    http请求头获取请求链接 File Request Links is a new feature we implemented which allows users to receive files ...

  9. python请求库_如何使用Python请求库发出post请求?

    我在Postman中使用以下过滤器在Web API中发出POST请求,但无法使用请求库在Python中发出简单的POST请求. 首先,我向这个URL(http://10.61.202.98:8081/ ...

最新文章

  1. Android SDK更新下载失败以及Studio首次安装取消自动下载SDK
  2. 遭遇一次MySQL猜解注入攻击
  3. 知云文献翻译打不开_文献翻译工具-知云文献翻译
  4. C++-十进制转二进制(matlab-dec2bin函数)
  5. TensorFlow2 实现神经风格迁移,DIY数字油画定制照片
  6. 一套不错的php,一套不错的PHP笔试题-附答案.pdf
  7. mac上qlv格式转mp4方法
  8. 结构化程序设计知识点总结
  9. 机器学习中的主动学习实现_我如何使用机器学习来帮助实现正念
  10. elasticsearch源码:unicast列表解析
  11. Linux常用命令速查手册——嘎嘎管用
  12. 终于懂了汇编代码为什么从键盘上输入字符,将该字符的ASCII显示在屏幕上必须要加30或37(附汇编代码)
  13. 远景能源面试+高管终面
  14. Kubernetes 基于ceph rbd生成pv
  15. 在fragment中引用ListView
  16. 哈工大操作系统课程实验记录
  17. 排队论(Queuing theory)简介
  18. mac去除dmg打开密码的方法
  19. 三菱Q程序,主站Q06UDE+QD75P4定位模块
  20. 【思维导图工具】万彩脑图大师教程 | 将思维导图分享到微信

热门文章

  1. R语言使用dplyr将特定的数据列移动到最前面、使用dplyr将特定数据列移动到另一指定数据列的后面、使用dplyr将特定数据列移动到另一指定数据列的前面
  2. R语言dataframe数据列格式转换(从整型integer转化为浮点型float)
  3. pandas为所有的列名添加后缀(add_suffix)
  4. R获取股票数据并进行进行可视化分析
  5. 为Jupyter notebook配置R kernel过程及踩坑记录
  6. 特征工程之时间特征、自然语言特征、图像图像特征、数据分布不平衡
  7. matplotlib画图、如何提高图像分辨率?
  8. 自定义服务器怎么调98k,《刺激战场》如何开自定义房间?升级可领房卡,3倍物资98k随便捡...
  9. Jabba: hybrid error correction for long sequencing reads using maximal exact matches机译:Jabba:使用最大精
  10. 查看linux系统版本信息 lsb_release -a