我是如何用 ThreadLocal 虐面试官的?,2024年最新java类的加载过程 面试题

先自我介绍一下,小编浙江大学毕业,去过华为、字节跳动等大厂,目前阿里P7

深知大多数程序员,想要提升技能,往往是自己摸索成长,但自己不成体系的自学效果低效又漫长,而且极易碰到天花板技术停滞不前!

因此收集整理了一份《2024年最新Java开发全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友。
img
img
img
img
img
img

既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,涵盖了95%以上Java开发知识点,真正体系化!

由于文件比较多,这里只是将部分目录截图出来,全套包含大厂面经、学习笔记、源码讲义、实战项目、大纲路线、讲解视频,并且后续会持续更新

如果你需要这些资料,可以添加V获取:vip1024b (备注Java)
img

正文

}

为什么 ThreadLocalMap 内部存储机构是维护一个数组呢?因为一个线程是可以通过多个不同的 ThreadLocal 对象来设置多个线程局部变量的,这些局部变量都是存储在自己线程的同一个 ThreadLocalMap 对象中。通过不同的 ThreadLocal 对象可以取得当前线程的不同局部变量值。

package com.chenpi;

/**

  • @Description

  • @Author 陈皮

  • @Date 2021/6/27

  • @Version 1.0

*/

public class ThreadLocalTest {

private static final ThreadLocal THREAD_LOCAL = new ThreadLocal<>();

private static final ThreadLocal THREAD_LOCAL01 = new ThreadLocal<>();

public static void main(String[] args) {

THREAD_LOCAL.set(“我是陈皮”);

System.out.println(THREAD_LOCAL.get());

THREAD_LOCAL01.set(“陈皮是我”);

System.out.println(THREAD_LOCAL01.get());

}

}

那同一个线程的 ThreadLocalMap 对象的数组 table,当前线程的不同 ThreadLocal 是如何确定数组下标,如果数组下标冲突又是怎么解决的呢?其实它不同于 HashMap 底层数组+链表+红黑树的存储结构,它只有 Entry 数组。

ThreadLocal 有个静态的初始哈希值 nextHashCode,然后每新建一个 ThreadLocal 对象都会在此哈希值的基础上自增一次,自增量为0x61c88647。

// 每 new 一个 ThreadLocal 对象都会自增一次哈希值

private final int threadLocalHashCode = nextHashCode();

// 初始哈希值,静态变量

private static AtomicInteger nextHashCode =

new AtomicInteger();

// 自增量

private static final int HASH_INCREMENT = 0x61c88647;

// 自增一次

private static int nextHashCode() {

return nextHashCode.getAndAdd(HASH_INCREMENT);

}

然后计算 table 数组下标是通过以下算法确定的,如果下标冲突,则下标会往后挪一位继续判断,直到不冲突为止。

// 首次创建 ThreadLocalMap 对象时,第一个元素的下标计算

int i = firstKey.threadLocalHashCode & (INITIAL_CAPACITY - 1);

// 后续元素的下标计算

int i = key.threadLocalHashCode & (len-1);

// 下标冲突时计算下一个下标的方法

private static int nextIndex(int i, int len) {

return ((i + 1 < len) ? i + 1 : 0);

}

我们看 ThreadLocal 类的 set 方法源码,它是设置线程局部变量的入口方法,实现原理也很简单。

  • 首先获取当前线程的 ThreadLocalMap 变量

  • 如果 ThreadLocalMap 变量存在,则将 ThreadLocal 对象和 T 数据以键值对的形式存储到 ThreadLocalMap 变量中

  • 如果 ThreadLocalMap 变量不存在,则新建 ThreadLocalMap 变量并绑定到当前线程中,再将 ThreadLocal 对象和 T 数据以键值对的形式存储到 ThreadLocalMap 变量中

// 设置线程局部变量

public void set(T value) {

Thread t = Thread.currentThread();

ThreadLocalMap map = getMap(t);

if (map != null)

map.set(this, value);

else

createMap(t, value);

}

ThreadLocal 类的 get 方法,它是访问线程局部变量的入口方法,实现原理也很简单。

  • 首先获取当前线程的 ThreadLocalMap 变量

  • 如果 ThreadLocalMap 变量存在,则将 ThreadLocal 对象作为 key,在 ThreadLocalMap 变量中查找对应的线程局部变量

  • 如果 ThreadLocalMap 变量不存在,则新建 ThreadLocalMap 变量并绑定到当前线程中,再将 ThreadLocal 对象和 null 以键值对的形式存储到 ThreadLocalMap 变量中

// 访问线程局部变量

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);

else

createMap(t, value);

return value;

}

protected T initialValue() {

return null;

}

ThreadLocal 类的 remove 方法,直接清除线程中 ThreadLocalMap 对象中以当前 ThreadLocal 对象为 key 的 Entry对象。

public void remove() {

ThreadLocalMap m = getMap(Thread.currentThread());

if (m != null)

m.remove(this);

}

你是否发现,ThreadLocal 类中的所有方法都是没有加锁的,因为 ThreadLocal 最终操作的都是对当前线程的 ThreadLocalMap 对象进行操作,既然线程处理自己的局部变量,就肯定不会有线程安全问题。

注意,同一个 ThreadLocal 变量在父线程中被设置值后,在子线程中是获取这个值的。即不具备继承性。具有继承性的是 InheritableThreadLocal 类,下期文章再讲解这个。

ThreadLocal 应用



ThreadLocal 具有线程隔离,线程安全的效果,如果数据是以线程为作用域并且不同线程具有不同的数据的时候,采用 ThreadLocal 是个不错的选择。

例如对于要用户登录的服务,对于每一个请求,我们可能需要校验用户是否登录,以及在登录后,后续的请求中会使用到用户信息,那我们就可以将登录校验过的用户信息放入线程局部变量中。

首先定义一个用户信息类,存放用户登录校验过的用户信息。

package com.chenpi;

import lombok.Data;

/**

  • @Description

  • @Author 陈皮

  • @Date 2021/6/27

  • @Version 1.0

*/

@Data

public class UserContext {

private String userId;

private String userName;

}

定义一个持有用户信息的管理工具类,主要用户管理当前线程的用户信息。

package com.chenpi;

/**

  • @Description

  • @Author 陈皮

  • @Date 2021/6/27

  • @Version 1.0

*/

public class UserContextHolder {

private static final ThreadLocal THREAD_LOCAL = new ThreadLocal<>();

private UserContextHolder() {}

public static void setUserContext(UserContext userContext) {

THREAD_LOCAL.set(userContext);

}

public static UserContext getUserContext() {

return THREAD_LOCAL.get();

}

public static void removeUserContext() {

THREAD_LOCAL.remove();

}

}

对需要用户权限的接口进行拦截,然后将用户信息存储到当前线程内部。注意,当请求完成后,需要将用户信息进行清除,避免内存泄露问题。

package com.chenpi;

import javax.servlet.http.HttpServletRequest;

import javax.servlet.http.HttpServletResponse;

import org.springframework.lang.Nullable;

import org.springframework.stereotype.Component;

import org.springframework.web.method.HandlerMethod;

import org.springframework.web.servlet.HandlerInterceptor;

/**

  • @Description 用户权限验证拦截

  • @Author 陈皮

  • @Date 2021/6/27

  • @Version 1.0

*/

@Component

public class UserPermissionInterceptor implements HandlerInterceptor {

@Override

public boolean preHandle(HttpServletRequest request, HttpServletResponse response,

Object handler) {

if (handler instanceof HandlerMethod) {

HandlerMethod handlerMethod = (HandlerMethod) handler;

// 获取用户权限校验注解

UserAuthenticate userAuthenticate =

handlerMethod.getMethod().getAnnotation(UserAuthenticate.class);

if (null == userAuthenticate) {

userAuthenticate = handlerMethod.getMethod().getDeclaringClass()

.getAnnotation(UserAuthenticate.class);

}

if (userAuthenticate != null && userAuthenticate.permission()) {

// 验证用户信息

UserContext userContext = userContextManager.getUserContext(request);

// 将用户信息存储到线程内部

UserContextHolder.setUserContext(userContext);

}

总结

蚂蚁面试比较重视基础,所以Java那些基本功一定要扎实。蚂蚁的工作环境还是挺赞的,因为我面的是稳定性保障部门,还有许多单独的小组,什么三年1班,很有青春的感觉。面试官基本水平都比较高,基本都P7以上,除了基础还问了不少架构设计方面的问题,收获还是挺大的。


经历这次面试我还通过一些渠道发现了需要大厂真实面试主要有:蚂蚁金服、拼多多、阿里云、百度、唯品会、携程、丰巢科技、乐信、软通动力、OPPO、银盛支付、中国平安等初,中级,高级Java面试题集合,附带超详细答案,希望能帮助到大家。

蚂蚁金服5面,总结了49个面试题,遇到的面试官都是P7级别以上

网上学习资料一大堆,但如果学到的知识不成体系,遇到问题时只是浅尝辄止,不再深入研究,那么很难做到真正的技术提升。

需要这份系统化的资料的朋友,可以添加V获取:vip1024b (备注Java)
img

一个人可以走的很快,但一群人才能走的更远!不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人,都欢迎加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!
实。蚂蚁的工作环境还是挺赞的,因为我面的是稳定性保障部门,还有许多单独的小组,什么三年1班,很有青春的感觉。面试官基本水平都比较高,基本都P7以上,除了基础还问了不少架构设计方面的问题,收获还是挺大的。


经历这次面试我还通过一些渠道发现了需要大厂真实面试主要有:蚂蚁金服、拼多多、阿里云、百度、唯品会、携程、丰巢科技、乐信、软通动力、OPPO、银盛支付、中国平安等初,中级,高级Java面试题集合,附带超详细答案,希望能帮助到大家。

[外链图片转存中…(img-kc0atOmw-1713131654601)]

网上学习资料一大堆,但如果学到的知识不成体系,遇到问题时只是浅尝辄止,不再深入研究,那么很难做到真正的技术提升。

需要这份系统化的资料的朋友,可以添加V获取:vip1024b (备注Java)
[外链图片转存中…(img-Df8Dk5ks-1713131654601)]

一个人可以走的很快,但一群人才能走的更远!不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人,都欢迎加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!

  • 25
    点赞
  • 8
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值