咱们能够经过把对象设计成不可变对象来躲避并发,咱们还能够经过使用线程封闭来实现线程安全,所谓线程封闭java
就是将数据都封装到一个线程里,不让其余线程访问。
ThreadLocal维护的是一个map 这个map是线程的名称多为key 咱们全部封闭的值做为value。
咱们作使用Filter作登陆操做都作过,咱们如今来使用ThreadLoad来存储一下用户信息。
public class RequestHolder { private final static ThreadLocal<Long> requestHolder = new ThreadLocal<>(); public static void add(Long id) { requestHolder.set(id); } public static Long getId() { return requestHolder.get(); } public static void remove() { requestHolder.remove(); } }
声明一个ThreadLoad对象用来存储ThreadLoad信息。安全
@Slf4j public class HttpFilter implements Filter { @Override public void init(FilterConfig filterConfig) throws ServletException { } @Override public void doFilter(ServletRequest servletRequest, ServletResponse servletResponse, FilterChain filterChain) throws IOException, ServletException { HttpServletRequest request = (HttpServletRequest) servletRequest; log.info("do filter, {}, {}", Thread.currentThread().getId(), request.getServletPath()); RequestHolder.add(Thread.currentThread().getId()); filterChain.doFilter(servletRequest, servletResponse); } @Override public void destroy() { }
@Slf4j public class HttpInterceptor extends HandlerInterceptorAdapter { @Override public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception { log.info("preHandle"); return true; } @Override public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) throws Exception { RequestHolder.remove(); log.info("afterCompletion"); return; } }
定义filter的内容,咱们在filter中将线程id加入到ThreadLoad中,而后在controller中获取,看看能不能获取的到。在线程执行完以后将ThreadLoad中的数据清空防止内存溢出。并发
@Controller @RequestMapping("/threadLocal") public class ThreadLocalController { @RequestMapping("/test") @ResponseBody public Long test() { return RequestHolder.getId(); } }
最后咱们用postman测试发现打印了线程id,ThreadLoad中变量值只要是一个线程中无论在哪一个类中都是共享的。app