最全并发编程十一java8新增的并发特性,java面试必看书籍

最后

这份清华大牛整理的进大厂必备的redis视频、面试题和技术文档

祝大家早日进入大厂,拿到满意的薪资和职级~~~加油!!

感谢大家的支持!!

image.png

本文已被CODING开源项目:【一线大厂Java面试题解析+核心总结学习笔记+最新讲解视频+实战项目源码】收录

需要这份系统化的资料的朋友,可以点击这里获取

/普通long的执行同步锁测试/

public void testSync() throws InterruptedException {

ExecutorService exe = Executors.newFixedThreadPool(MAX_THREADS);

long starttime = System.currentTimeMillis();

SyncTask sync = new SyncTask(starttime, this);

for (int i = 0; i < TASK_COUNT; i++) {

exe.submit(sync);

}

cdlsync.await();

exe.shutdown();

}

/原子型long的测试任务/

public class AtomicTask implements Runnable {

protected String name;

protected long starttime;

public AtomicTask(long starttime) {

this.starttime = starttime;

}

@Override

public void run() {

long v = acount.get();

while (v < TARGET_COUNT) {

v = acount.incrementAndGet();

}

long endtime = System.currentTimeMillis();

System.out.println(“AtomicTask spend:” + (endtime - starttime) + “ms” );

cdlatomic.countDown();

}

}

/原子型long的执行测试/

public void testAtomic() throws InterruptedException {

ExecutorService exe = Executors.newFixedThreadPool(MAX_THREADS);

long starttime = System.currentTimeMillis();

AtomicTask atomic = new AtomicTask(starttime);

for (int i = 0; i < TASK_COUNT; i++) {

exe.submit(atomic);

}

cdlatomic.await();

exe.shutdown();

}

/LongAdder的测试任务/

public class LongAdderTask implements Runnable {

protected String name;

protected long startTime;

public LongAdderTask(long startTime) {

this.startTime = startTime;

}

@Override

public void run() {

long v = lacount.sum();

while (v < TARGET_COUNT) {

lacount.increment();

v = lacount.sum();

}

long endtime = System.currentTimeMillis();

System.out.println(“LongAdderTask spend:” + (endtime - startTime) + “ms”);

cdladdr.countDown();

}

}

/LongAdder的执行测试/

public void testLongAdder() throws InterruptedException {

ExecutorService exe = Executors.newFixedThreadPool(MAX_THREADS);

long startTime = System.currentTimeMillis();

LongAdderTask longAdderTask = new LongAdderTask(startTime);

for (int i = 0; i < TASK_COUNT; i++) {

exe.submit(longAdderTask);

}

cdladdr.await();

exe.shutdown();

}

public static void main(String[] args) throws InterruptedException {

LongAdderDemo demo = new LongAdderDemo();

demo.testSync();

demo.testAtomic();

demo.testLongAdder();

}

}

二、StampLock


StampedLock是Java8引入的一种新的所机制,简单的理解,可以认为它是读写锁的一个改进版本,读写锁虽然分离了读和写的功能,使得读与读之间可以完全并发,但是读和写之间依然是冲突的,读锁会完全阻塞写锁,它使用的依然是悲观的锁策略.如果有大量的读线程,他也有可能引起写线程的饥饿。

而StampedLock则提供了一种乐观的读策略,这种乐观策略的锁非常类似于无锁的操作,使得乐观锁完全不会阻塞写线程。

它的思想是读写锁中读不仅不阻塞读,同时也不应该阻塞写。

读不阻塞写的实现思路:

在读的时候如果发生了写,则应当重读而不是在读的时候直接阻塞写!即读写之间不会阻塞对方,但是写和写之间还是阻塞的!

StampedLock的内部实现是基于CLH的。

code:

import java.util.concurrent.locks.StampedLock;

/**

  • 类说明:JDK1.8源码自带的示例

*/

public class StampedLockDemo {

//一个点的x,y坐标

private double x,y;

/**Stamped类似一个时间戳的作用,每次写的时候对其+1来改变被操作对象的Stamped值

  • 这样其它线程读的时候发现目标对象的Stamped改变,则执行重读*/

private final StampedLock sl = new StampedLock();

//【写锁(排它锁)】

void move(double deltaX,double deltaY) {// an exclusively locked method

/**stampedLock调用writeLock和unlockWrite时候都会导致stampedLock的stamp值的变化

  • 即每次+1,直到加到最大值,然后从0重新开始*/

long stamp =sl.writeLock(); //写锁

try {

x +=deltaX;

y +=deltaY;

} finally {

sl.unlockWrite(stamp);//释放写锁

}

}

//【乐观读锁】

double distanceFromOrigin() { // A read-only method

/**

  • tryOptimisticRead是一个乐观的读,使用这种锁的读不阻塞写

  • 每次读的时候得到一个当前的stamp值(类似时间戳的作用)

*/

long stamp = sl.tryOptimisticRead();

//这里就是读操作,读取x和y,因为读取x时,y可能被写了新的值,所以下面需要判断

double currentX = x, currentY = y;

/**如果读取的时候发生了写,则stampedLock的stamp属性值会变化,此时需要重读,

  • validate():比较当前stamp和获取乐观锁得到的stamp比较,不一致则失败。

  • 再重读的时候需要加读锁(并且重读时使用的应当是悲观的读锁,即阻塞写的读锁)

  • 当然重读的时候还可以使用tryOptimisticRead,此时需要结合循环了,即类似CAS方式

  • 读锁又重新返回一个stampe值*/

if (!sl.validate(stamp)) {//如果验证失败(读之前已发生写)

stamp = sl.readLock(); //悲观读锁

try {

currentX = x;

currentY = y;

}finally{

sl.unlockRead(stamp);//释放读锁

}

}

//读锁验证成功后执行计算,即读的时候没有发生写

return Math.sqrt(currentX *currentX + currentY *currentY);

}

//读锁升级为写锁

void moveIfAtOrigin(double newX, double newY) { // upgrade

// 读锁(这里可用乐观锁替代)

long stamp = sl.readLock();

try {

//循环,检查当前状态是否符合

while (x == 0.0 && y == 0.0) {

long ws = sl.tryConvertToWriteLock(stamp);

//如果写锁成功

if (ws != 0L) {

stamp = ws;// 替换stamp为写锁戳

x = newX;//修改数据

y = newY;

break;

}

//转换为写锁失败

else {

//释放读锁

sl.unlockRead(stamp);

//获取写锁(必要情况下阻塞一直到获取写锁成功)

stamp = sl.writeLock();

}

}

} finally {

//释放锁(可能是读/写锁)

sl.unlock(stamp);

}

}

}

三、CompleteableFuture


Future的不足

Future是Java 5添加的类,用来描述一个异步计算的结果。你可以使用isDone方法检查计算是否完成,或者使用get阻塞住调用线程,直到计算完成返回结果,你也可以使用cancel方法停止任务的执行。

虽然Future以及相关使用方法提供了异步执行任务的能力,但是对于结果的获取却是很不方便,只能通过阻塞或者轮询的方式得到任务的结果。阻塞的方式显然和我们的异步编程的初衷相违背,轮询的方式又会耗费无谓的CPU资源,而且也不能及时地得到计算结果,为什么不能用观察者设计模式当计算结果完成及时通知监听者呢?。

Java的一些框架,比如Netty,自己扩展了Java的 Future接口,提供了addListener等多个扩展方法,Google guava也提供了通用的扩展Future:ListenableFuture、SettableFuture 以及辅助类Futures等,方便异步编程。

同时Future接口很难直接表述多个Future 结果之间的依赖性。实际开发中,我们经常需要达成以下目的:

将两个异步计算合并为一个——这两个异步计算之间相互独立,同时第二个又依赖于第一个的结果。

等待 Future 集合中的所有任务都完成。

仅等待 Future集合中最快结束的任务完成(有可能因为它们试图通过不同的方式计算同一个值),并返回它的结果。

应对 Future 的完成事件(即当 Future 的完成事件发生时会收到通知,并能使用 Future 计算的结果进行下一步的操作,不只是简单地阻塞等待操作的结果)

CompleteableFuture

JDK1.8才新加入的一个实现类CompletableFuture,实现了Future, CompletionStage两个接口。实现了Future接口,意味着可以像以前一样通过阻塞或者轮询的方式获得结果。

创建

除了直接new出一个CompletableFuture的实例,还可以通过工厂方法创建CompletableFuture的实例

工厂方法:

Asynsc表示异步,而supplyAsync与runAsync不同在与前者异步返回一个结果,后者是void.第二个函数第二个参数表示是用我们自己创建的线程池,否则采用默认的ForkJoinPool.commonPool()作为它的线程池。

获得结果的方法

public T get()

public T get(long timeout, TimeUnit unit)

public T getNow(T valueIfAbsent)

public T join()

getNow有点特殊,如果结果已经计算完则返回结果或者抛出异常,否则返回给定的valueIfAbsent值。

join返回计算的结果或者抛出一个unchecked异常(CompletionException),它和get对抛出的异常的处理有些细微的区别。

辅助方法

public static CompletableFuture allOf(CompletableFuture<?>… cfs)

public static CompletableFuture anyOf(CompletableFuture<?>… cfs)

allOf方法是当所有的CompletableFuture都执行完后执行计算。

anyOf方法是当任意一个CompletableFuture执行完后就会执行计算,计算的结果相同。

CompletionStage是一个接口,从命名上看得知是一个完成的阶段,它代表了一个特定的计算的阶段,可以同步或者异步的被完成。你可以把它看成一个计算流水线上的一个单元,并最终会产生一个最终结果,这意味着几个CompletionStage可以串联起来,一个完成的阶段可以触发下一阶段的执行,接着触发下一次,再接着触发下一次,……….。

总结CompletableFuture几个关键点:

1、计算可以由 Future ,Consumer 或者 Runnable 接口中的 apply,accept 或者 run等方法表示。

2、计算的执行主要有以下

a. 默认执行

b. 使用默认的CompletionStage的异步执行提供者异步执行。这些方法名使用someActionAsync这种格式表示。

c. 使用 Executor 提供者异步执行。这些方法同样也是someActionAsync这种格式,但是会增加一个Executor 参数。

CompletableFuture里大约有五十种方法,但是可以进行归类

变换类 thenApply:

关键入参是函数式接口Function。它的入参是上一个阶段计算后的结果,返回值是经过转化后结果。

消费类 thenAccept:

关键入参是函数式接口Consumer。它的入参是上一个阶段计算后的结果, 没有返回值。

执行操作类 thenRun:

对上一步的计算结果不关心,执行下一个操作,入参是一个Runnable的实例,表示上一步完成后执行的操作。

结合转化类:

需要上一步的处理返回值,并且other代表的CompletionStage 有返回值之后,利用这两个返回值,进行转换后返回指定类型的值。

两个CompletionStage是并行执行的,它们之间并没有先后依赖顺序,other并不会等待先前的CompletableFuture执行完毕后再执行。

结合转化类

对于Compose可以连接两个CompletableFuture,其内部处理逻辑是当第一个CompletableFuture处理没有完成时会合并成一个CompletableFuture,如果处理完成,第二个future会紧接上一个CompletableFuture进行处理。

第一个CompletableFuture 的处理结果是第二个future需要的输入参数。

结合消费类:

需要上一步的处理返回值,并且other代表的CompletionStage 有返回值之后,利用这两个返回值,进行消费

运行后执行类:

不关心这两个CompletionStage的结果,只关心这两个CompletionStage都执行完毕,之后再进行操作(Runnable)。

取最快转换类:

两个CompletionStage,谁计算的快,我就用那个CompletionStage的结果进行下一步的转化操作。现实开发场景中,总会碰到有两种渠道完成同一个事情,所以就可以调用这个方法,找一个最快的结果进行处理。

取最快消费类:

两个CompletionStage,谁计算的快,我就用那个CompletionStage的结果进行下一步的消费操作。

取最快运行后执行类:

两个CompletionStage,任何一个完成了都会执行下一步的操作(Runnable)。

异常补偿类:

当运行时出现了异常,可以通过exceptionally进行补偿。

运行后记录结果类:

action执行完毕后它的结果返回原始的CompletableFuture的计算结果或者返回异常。所以不会对结果产生任何的作用。

运行后处理结果类:

运行完成时,对结果的处理。这里的完成时有两种情况,一种是正常执行,返回值。另外一种是遇到异常抛出造成程序的中断。

四、扩充知识点- Disruptor

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

应用背景和介绍


Disruptor是英国外汇交易公司LMAX开发的一个高性能队列,研发的初衷是解决内部的内存队列的延迟问题,而不是分布式队列。基于Disruptor开发的系统单线程能支撑每秒600万订单,2010年在QCon演讲后,获得了业界关注。

据目前资料显示:应用Disruptor的知名项目有如下的一些:Storm, Camel, Log4j2,还有目前的美团点评技术团队也有很多不少的应用,或者说有一些借鉴了它的设计机制。

Disruptor是一个高性能的线程间异步通信的框架,即在同一个JVM进程中的多线程间消息传递。

传统队列问题


在JDK中,Java内部的队列BlockQueue的各种实现,仔细分析可以得知,队列的底层数据结构一般分成三种:数组、链表和堆,堆这里是为了实现带有优先级特性的队列暂且不考虑。

在稳定性和性能要求特别高的系统中,为了防止生产者速度过快,导致内存溢出,只能选择有界队列;同时,为了减少Java的垃圾回收对系统性能的影响,会尽量选择 Array格式的数据结构。这样筛选下来,符合条件的队列就只有ArrayBlockingQueue。但是ArrayBlockingQueue是通过加锁的方式保证线程安全,而且ArrayBlockingQueue还存在伪共享问题,这两个问题严重影响了性能。

ArrayBlockingQueue的这个伪共享问题存在于哪里呢,分析下核心的部分源码,其中最核心的三个成员变量为

是在ArrayBlockingQueue的核心enqueue和dequeue方法中经常会用到的,这三个变量很容易放到同一个缓存行中,进而产生伪共享问题。

高性能的原理


引入环形的数组结构:数组元素不会被回收,避免频繁的GC,

无锁的设计:采用CAS无锁方式,保证线程的安全性

属性填充:通过添加额外的无用信息,避免伪共享问题

环形数组结构是整个Disruptor的核心所在。

首先因为是数组,所以要比链表快,而且根据我们对上面缓存行的解释知道,数组中的一个元素加载,相邻的数组元素也是会被预加载的,因此在这样的结构中,cpu无需时不时去主存加载数组中的下一个元素。而且,你可以为数组预先分配内存,使得数组对象一直存在(除非程序终止)。这就意味着不需要花大量的时间用于垃圾回收。此外,不像链表那样,需要为每一个添加到其上面的对象创造节点对象—对应的,当删除节点时,需要执行相应的内存清理操作。环形数组中的元素采用覆盖方式,避免了jvm的GC。

其次结构作为环形,数组的大小为2的n次方,这样元素定位可以通过位运算效率会更高,这个跟一致性哈希中的环形策略有点像。在disruptor中,这个牛逼的环形结构就是RingBuffer,既然是数组,那么就有大小,而且这个大小必须是2的n次方

其实质只是一个普通的数组,只是当放置数据填充满队列(即到达2^n-1位置)之后,再填充数据,就会从0开始,覆盖之前的数据,于是就相当于一个环。

每个生产者首先通过CAS竞争获取可以写的空间,然后再进行慢慢往里放数据,如果正好这个时候消费者要消费数据,那么每个消费者都需要获取最大可消费的下标。

同时,Disruptor 不像传统的队列,分为一个队头指针和一个队尾指针,而是只有一个角标(上图的seq),它属于一个volatile变量,同时也是我们能够不用锁操作就能实现Disruptor的原因之一,而且通过缓存行补充,避免伪共享问题。该指针是通过一直自增的方式来获取下一个可写或者可读数据。

本章主要讲了LongAdder,StampLock,CompleteableFuture等java8中新增的一些特性,以及简单介绍了Disruptor,希望大家在开发中可以有更多的选择。

其他阅读

写在最后

很多人感叹“学习无用”,实际上之所以产生无用论,是因为自己想要的与自己所学的匹配不上,这也就意味着自己学得远远不够。无论是学习还是工作,都应该有主动性,所以如果拥有大厂梦,那么就要自己努力去实现它。

最后祝愿各位身体健康,顺利拿到心仪的offer!

由于文章的篇幅有限,所以这次的蚂蚁金服和京东面试题答案整理在了PDF文档里

蚂蚁、京东Java岗4面:原理+索引+底层+分布式+优化等,已拿offer

蚂蚁、京东Java岗4面:原理+索引+底层+分布式+优化等,已拿offer

蚂蚁、京东Java岗4面:原理+索引+底层+分布式+优化等,已拿offer

本文已被CODING开源项目:【一线大厂Java面试题解析+核心总结学习笔记+最新讲解视频+实战项目源码】收录

需要这份系统化的资料的朋友,可以点击这里获取

个时候消费者要消费数据,那么每个消费者都需要获取最大可消费的下标。

同时,Disruptor 不像传统的队列,分为一个队头指针和一个队尾指针,而是只有一个角标(上图的seq),它属于一个volatile变量,同时也是我们能够不用锁操作就能实现Disruptor的原因之一,而且通过缓存行补充,避免伪共享问题。该指针是通过一直自增的方式来获取下一个可写或者可读数据。

本章主要讲了LongAdder,StampLock,CompleteableFuture等java8中新增的一些特性,以及简单介绍了Disruptor,希望大家在开发中可以有更多的选择。

其他阅读

写在最后

很多人感叹“学习无用”,实际上之所以产生无用论,是因为自己想要的与自己所学的匹配不上,这也就意味着自己学得远远不够。无论是学习还是工作,都应该有主动性,所以如果拥有大厂梦,那么就要自己努力去实现它。

最后祝愿各位身体健康,顺利拿到心仪的offer!

由于文章的篇幅有限,所以这次的蚂蚁金服和京东面试题答案整理在了PDF文档里

[外链图片转存中…(img-EwfJ6Prp-1715602198709)]

[外链图片转存中…(img-hVeaR8GF-1715602198710)]

[外链图片转存中…(img-d7gbqtzh-1715602198710)]

本文已被CODING开源项目:【一线大厂Java面试题解析+核心总结学习笔记+最新讲解视频+实战项目源码】收录

需要这份系统化的资料的朋友,可以点击这里获取

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

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值