一次性搞清楚,Java并发编程在各主流框架中的应用,保证看懂

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

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

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

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

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

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

正文

return id;

}

public void setId(String id) {

this.id = id;

}

}

public void createSession() {

session.set(new Session());

}

public void setId(String id) {

session.get().setId(id);

}

public String getId() {

return session.get().getId();

}

public static void main(String[] args) {

new Thread(() -> {

SessionBean bean = new SessionBean();

bean.createSession();

bean.setId(“susan”);

System.out.println(bean.getId());

}).start();

}

}

在方法的内部实现中,直接可以通过 session.get() 获取到当前线程的 session,省掉了参数在方法间传递的环节。

ThreadLocal 的实现原理

一般,类属性中的数据是多个线程共享的,但 ThreadLocal 类型的数据 声明为类属性,却可以为每一个使用它(通过 set(T value)方法)的线程存储线程私有的数据,通过其源码我们可以发现其中的原理。

public class ThreadLocal {

/**

  • 下面的 getMap()方法 传入当前线程,获得一个ThreadLocalMap对象,说明每一个线程维护了

  • 自己的一个 map,保证读取出来的value是自己线程的。

  • ThreadLocalMap 是ThreadLocal静态内部类,存储value的键值就是ThreadLocal本身。

  • 因此可以断定,每个线程维护一个ThreadLocalMap的键值对映射Map。不同线程的Map的 key值 是一样的,

  • 都是ThreadLocal,但 value 是不同的。

*/

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

} 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 在 Spring 中的使用

Spring 事务处理的设计与实现中大量使用了 ThreadLocal 类,比如,TransactionSynchronizationManager 维护了一系列的 ThreadLocal 变量,用于存储线程私有的 事务属性及资源。源码如下。

/**

  • 管理每个线程的资源和事务同步的中心帮助程序。供资源管理代码使用,但不供典型应用程序代码使用。

  • 资源管理代码应该检查线程绑定的资源,如,JDBC连接 或 Hibernate Sessions。

  • 此类代码通常不应该将资源绑定到线程,因为这是事务管理器的职责。另一个选项是,

  • 如果事务同步处于活动状态,则在首次使用时延迟绑定,以执行跨任意数量资源的事务。

*/

public abstract class TransactionSynchronizationManager {

/**

  • 一般是一个线程持有一个 独立的事务,以相互隔离地处理各自的事务。

  • 所以这里使用了很多 ThreadLocal对象,为每个线程绑定 对应的事务属性及资源,

  • 以便后续使用时能直接获取。

*/

private static final ThreadLocal<Map<Object, Object>> resources =

new NamedThreadLocal<Map<Object, Object>>(“Transactional resources”);

private static final ThreadLocal<Set> synchronizations =

new NamedThreadLocal<Set>(“Transaction synchronizations”);

private static final ThreadLocal currentTransactionName =

new NamedThreadLocal(“Current transaction name”);

private static final ThreadLocal currentTransactionReadOnly =

new NamedThreadLocal(“Current transaction read-only status”);

private static final ThreadLocal currentTransactionIsolationLevel =

new NamedThreadLocal(“Current transaction isolation level”);

private static final ThreadLocal actualTransactionActive =

new NamedThreadLocal(“Actual transaction active”);

/**

  • 为当前线程 绑定 对应的resource资源

*/

public static void bindResource(Object key, Object value) throws IllegalStateException {

Object actualKey = TransactionSynchronizationUtils.unwrapResourceIfNecessary(key);

Assert.notNull(value, “Value must not be null”);

Map<Object, Object> map = resources.get();

// 如果当前线程的 resources中,绑定的数据map为空,则为 resources 绑定 map

if (map == null) {

map = new HashMap<Object, Object>();

resources.set(map);

}

Object oldValue = map.put(actualKey, value);

if (oldValue instanceof ResourceHolder && ((ResourceHolder) oldValue).isVoid()) {

oldValue = null;

}

if (oldValue != null) {

throw new IllegalStateException(“Already value [” + oldValue + “] for key [” +

actualKey + “] bound to thread [” + Thread.currentThread().getName() + “]”);

}

if (logger.isTraceEnabled()) {

logger.trace(“Bound value [” + value + “] for key [” + actualKey + “] to thread [” +

Thread.currentThread().getName() + “]”);

}

}

/**

  • 返回当前线程绑定的所有资源

*/

public static Map<Object, Object> getResourceMap() {

Map<Object, Object> map = resources.get();

return (map != null ? Collections.unmodifiableMap(map) : Collections.emptyMap());

}

}

ThreadLocal 在 Mybatis 中的使用

Mybatis 的 SqlSession 对象 也是各线程私有的资源,所以对其的管理也使用到了 ThreadLocal 类。源码如下。

public class SqlSessionManager implements SqlSessionFactory, SqlSession {

private final ThreadLocal localSqlSession = new ThreadLocal<>();

public void startManagedSession() {

this.localSqlSession.set(openSession());

} public void startManagedSession(boolean autoCommit) {

this.localSqlSession.set(openSession(autoCommit));

} public void startManagedSession(Connection connection) {

this.localSqlSession.set(openSession(connection));

} public void startManagedSession(TransactionIsolationLevel level) {

this.localSqlSession.set(openSession(level));

} public void startManagedSession(ExecutorType execType) {

this.localSqlSession.set(openSession(execType));

} public void startManagedSession(ExecutorType execType, boolean autoCommit) {

this.localSqlSession.set(openSession(execType, autoCommit));

} public void startManagedSession(ExecutorType execType, TransactionIsolationLevel level) {

this.localSqlSession.set(openSession(execType, level));

} public void startManagedSession(ExecutorType execType, Connection connection) {

this.localSqlSession.set(openSession(execType, connection));

} public boolean isManagedSessionStarted() {

return this.localSqlSession.get() != null;

} @Override

public Connection getConnection() {

final SqlSession sqlSession = localSqlSession.get();

if (sqlSession == null) {

throw new SqlSessionException(“Error: Cannot get connection. No managed session is started.”);

} return sqlSession.getConnection();

} @Override

public void clearCache() {

final SqlSession sqlSession = localSqlSession.get();

if (sqlSession == null) {

throw new SqlSessionException(“Error: Cannot clear the cache. No managed session is started.”);

} sqlSession.clearCache(); } @Override

public void commit() {

final SqlSession sqlSession = localSqlSession.get();

if (sqlSession == null) {

throw new SqlSessionException(“Error: Cannot commit. No managed session is started.”);

} sqlSession.commit(); } @Override

public void commit(boolean force) {

final SqlSession sqlSession = localSqlSession.get();

if (sqlSession == null) {

throw new SqlSessionException(“Error: Cannot commit. No managed session is started.”);

} sqlSession.commit(force); } @Override

public void rollback() {

final SqlSession sqlSession = localSqlSession.get();

if (sqlSession == null) {

throw new SqlSessionException(“Error: Cannot rollback. No managed session is started.”);

} sqlSession.rollback(); } @Override

public void rollback(boolean force) {

final SqlSession sqlSession = localSqlSession.get();

if (sqlSession == null) {

throw new SqlSessionException(“Error: Cannot rollback. No managed session is started.”);

} sqlSession.rollback(force); } @Override

public List flushStatements() {

final SqlSession sqlSession = localSqlSession.get();

if (sqlSession == null) {

throw new SqlSessionException(“Error: Cannot rollback. No managed session is started.”);

} return sqlSession.flushStatements();

} @Override

public void close() {

final SqlSession sqlSession = localSqlSession.get();

if (sqlSession == null) {

throw new SqlSessionException(“Error: Cannot close. No managed session is started.”);

} try {

sqlSession.close(); } finally {

localSqlSession.set(null);

} }}

J.U.C 包的实际应用

================

线程池 ThreadPoolExecutor

首先通过 ThreadPoolExecutor 的源码 看一下线程池的主要参数及方法。

public class ThreadPoolExecutor extends AbstractExecutorService {

/**

  • 核心线程数

  • 当向线程池提交一个任务时,若线程池已创建的线程数小于corePoolSize,即便此时存在空闲线程,

  • 也会通过创建一个新线程来执行该任务,直到已创建的线程数大于或等于corePoolSize

*/

private volatile int corePoolSize;

/**

  • 最大线程数

  • 当队列满了,且已创建的线程数小于maximumPoolSize,则线程池会创建新的线程来执行任务。

  • 另外,对于无界队列,可忽略该参数

*/

private volatile int maximumPoolSize;

/**

  • 线程存活保持时间

  • 当线程池中线程数 超出核心线程数,且线程的空闲时间也超过 keepAliveTime时,

  • 那么这个线程就会被销毁,直到线程池中的线程数小于等于核心线程数

*/

private volatile long keepAliveTime;

/**

  • 任务队列

  • 用于传输和保存等待执行任务的阻塞队列

*/

private final BlockingQueue workQueue;

/**

  • 线程工厂

  • 用于创建新线程。threadFactory 创建的线程也是采用 new Thread() 方式,threadFactory

  • 创建的线程名都具有统一的风格:pool-m-thread-n(m为线程池的编号,n为线程池中线程的编号

*/

private volatile ThreadFactory threadFactory;

/**

  • 线程饱和策略

  • 当线程池和队列都满了,再加入的线程会执行此策略

*/

private volatile RejectedExecutionHandler handler;

/**

  • 构造方法提供了多种重载,但实际上都使用了最后一个重载 完成了实例化

*/

public ThreadPoolExecutor(int corePoolSize,

int maximumPoolSize,

long keepAliveTime,

TimeUnit unit,

BlockingQueue workQueue) {

this(corePoolSize, maximumPoolSize, keepAliveTime, unit, workQueue,

Executors.defaultThreadFactory(), defaultHandler); } public ThreadPoolExecutor(int corePoolSize,

int maximumPoolSize,

long keepAliveTime,

TimeUnit unit,

BlockingQueue workQueue,

ThreadFactory threadFactory) {

this(corePoolSize, maximumPoolSize, keepAliveTime, unit, workQueue,

threadFactory, defaultHandler); } public ThreadPoolExecutor(int corePoolSize,

int maximumPoolSize,

long keepAliveTime,

TimeUnit unit,

BlockingQueue workQueue,

RejectedExecutionHandler handler) {

this(corePoolSize, maximumPoolSize, keepAliveTime, unit, workQueue,

Executors.defaultThreadFactory(), handler); } public ThreadPoolExecutor(int corePoolSize,

int maximumPoolSize,

long keepAliveTime,

TimeUnit unit,

BlockingQueue workQueue,

ThreadFactory threadFactory,

RejectedExecutionHandler handler) {

if (corePoolSize < 0 ||

maximumPoolSize <= 0 ||

maximumPoolSize < corePoolSize || keepAliveTime < 0)

throw new IllegalArgumentException();

if (workQueue == null || threadFactory == null || handler == null)

throw new NullPointerException();

this.corePoolSize = corePoolSize;

this.maximumPoolSize = maximumPoolSize;

this.workQueue = workQueue;

this.keepAliveTime = unit.toNanos(keepAliveTime);

this.threadFactory = threadFactory;

this.handler = handler;

} /**

  • 执行一个任务,但没有返回值

*/

public void execute(Runnable command) {

if (command == null)

throw new NullPointerException();

int c = ctl.get();

if (workerCountOf© < corePoolSize) {

if (addWorker(command, true))

return;

c = ctl.get();

} if (isRunning© && workQueue.offer(command)) {

int recheck = ctl.get();

if (! isRunning(recheck) && remove(command))

reject(command); else if (workerCountOf(recheck) == 0)

addWorker(null, false);

} else if (!addWorker(command, false))

reject(command); } /**

  • 提交一个线程任务,有返回值。该方法继承自其父类 AbstractExecutorService,有多种重载,这是最常用的一个。

  • 通过future.get()获取返回值(阻塞直到任务执行完)

*/

public Future submit(Callable task) {

if (task == null) throw new NullPointerException();

RunnableFuture ftask = newTaskFor(task); execute(ftask); return ftask;

} /**

  • 关闭线程池,不再接收新的任务,但会把已有的任务执行完

*/

public void shutdown() {

final ReentrantLock mainLock = this.mainLock;

mainLock.lock();

try {

checkShutdownAccess(); advanceRunState(SHUTDOWN); interruptIdleWorkers(); onShutdown(); // hook for ScheduledThreadPoolExecutor

} finally {

mainLock.unlock();

}

tryTerminate();

}

/**

  • 立即关闭线程池,已有的任务也会被抛弃

*/

public List shutdownNow() {

List tasks;

final ReentrantLock mainLock = this.mainLock;

mainLock.lock();

try {

checkShutdownAccess();

advanceRunState(STOP);

interruptWorkers();

tasks = drainQueue();

} finally {

mainLock.unlock();

}

tryTerminate();

return tasks;

Kafka实战笔记

关于这份笔记,为了不影响大家的阅读体验,我只能在文章中展示部分的章节内容和核心截图

image.png

  • Kafka入门
  • 为什么选择Kafka
  • Karka的安装、管理和配置

image.png

  • Kafka的集群
  • 第一个Kafka程序
  • image.png

afka的生产者

image.png

  • Kafka的消费者
  • 深入理解Kafka
  • 可靠的数据传递

image.png

image.png

  • Spring和Kalka的整合
  • Sprinboot和Kafka的整合
  • Kafka实战之削峰填谷
  • 数据管道和流式处理(了解即可)

image.png

  • Kafka实战之削峰填谷

image.png

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

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

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

第一个Kafka程序

  • [外链图片转存中…(img-tswzbwuB-1713672707382)]

afka的生产者

[外链图片转存中…(img-3xuB130w-1713672707383)]

  • Kafka的消费者
  • 深入理解Kafka
  • 可靠的数据传递

[外链图片转存中…(img-SonuWtP8-1713672707384)]

[外链图片转存中…(img-TxyJH12m-1713672707384)]

  • Spring和Kalka的整合
  • Sprinboot和Kafka的整合
  • Kafka实战之削峰填谷
  • 数据管道和流式处理(了解即可)

[外链图片转存中…(img-LmMhXLpS-1713672707385)]

  • Kafka实战之削峰填谷

[外链图片转存中…(img-lx0j0Dm9-1713672707386)]

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

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

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

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

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值