干货来袭,吐血整理出来的JAVA之JUC详解,还不赶紧点击观摩观摩!

本文详述了Java并发编程中重要的JUC(Java Util Concurrency)工具包,包括volatile关键字的特性、i++的原子性问题、CAS算法、并发容器如ConcurrentHashMap、CountDownLatch和线程池的实现与使用。通过理解这些概念,开发者能更好地掌握多线程环境下的高效编程。
摘要由CSDN通过智能技术生成

1. JUC 简介

在 Java 5.0 提供了 java.util.concurrent(简称JUC)包,在此包中增加了在并发编程中很常用的工具类,
用于定义类似于线程的自定义子系统,包括线程池,异步 IO 和轻量级任务框架;还提供了设计用于多线程上下文中的 Collection 实现等;

2. volatile 关键字

volatile 关键字: 当多个线程进行操作共享数据时,可以保证内存中的数据是可见的;相较于 synchronized 是一种
较为轻量级的同步策略;
volatile 不具备"互斥性";
volatile 不能保证变量的"原子性";

// 使用 volatile 之前
public class TestVolatile{
   

    public static void main(String[] args){
   
        ThreadDemo td = new ThreadDemo();
        new Thread(td).start();

        while(true){
   
            if(td.isFlag()){
   
                System.out.println("########");
                break;
            }
        }
    }
}

class ThreadDemo implements Runnable{
   
    private boolean flag = false;

    public void run(){
   
        try{
   
            // 该线程 sleep(200), 导致了程序无法执行成功
            Thread.sleep(200);
        }catch(InterruptedException e){
   
            e.printStackTrace();
        }

        flag = true;

        Sytem.out.println("flag="+isFlag());
    }

    public boolean isFlag(){
   
        return flag;
    }

    public void setFlag(boolean flag){
   
        this.flag = flag;
    }
}

在这里插入图片描述
在这里插入图片描述

// 解决问题方式一: 同步锁
//   但是,效率太低
public class TestVolatile{
   

    public static void main(String[] args){
   
        ThreadDemo td = new ThreadDemo();
        new Thread(td).start();


        while(true){
   
            // 使用同步锁
            synchronized(td){
   
                if(td.isFlag()){
   
                    System.out.println("########");
                    break;
                }
            }
        }
    }
}

// 解决方式二: 使用 volatile 关键字
public class TestVolatile{
   

    public static void main(String[] args){
   
        ThreadDemo td = new ThreadDemo();
        new Thread(td).start();

        while(true){
   
            if(td.isFlag()){
   
                System.out.println("########");
                break;
            }
        }
    }
}

class ThreadDemo implements Runnable{
   
    private volatile boolean flag = false;

    同上()
}

3. i++ 的原子性问题

1.i++的操作实际上分为三个步骤: “读-改-写”;

2.原子性: 就是"i++"的"读-改-写"是不可分割的三个步骤;

3.原子变量: JDK1.5 以后, java.util.concurrent.atomic包下,提供了常用的原子变量;

原子变量中的值,使用 volatile 修饰,保证了内存可见性;
CAS(Compare-And-Swap) 算法保证数据的原子性;

int i = 10;
i = i++;  // 此时, i=10

执行步骤:
int temp = i;
i = i + 1;
i = temp;


// 测试类
public class TestAtomicDemo{
   
    public static void main(String[] args){
   

        AtomicDemo ad = new AtomicDemo();

        for(int i=0; i < 10; i++){
   
            new Thread(ad).start();
        }
    }
}

class AtomicDemo implements Runnable{
   
    private int serialNumber = 0;

    public void run(){
   

        try{
   
            Thread.sleep(200);
        }catch(InterruptedException e){
   

        }

        System.out.println(Thread.currentThread().getName() + ":" + getSerialNumber());
    }

    public int getSerialNumber(){
   
        return serialNumber++;
    }
}

在这里插入图片描述

// 改进: 使用原子变量
class AtomicDemo implements Runnable{
   

    private AtomicInteger serialNumber = new AtomicInteger();

    public void run(){
   
        try{
   
            Thread.sleep(200);
        }catch(InterruptedException e){
   

        }

        System.out.println(Thread.currentThread().getName()+":"+getSerialNumber());
    }

    public int getSerialNumber(){
   
        // 自增运算
        return serialNumber.getAndIncrement();
    }
}

3.1 CAS 算法

CAS(Compare-And-Swap) 算法是硬件对于并发的支持,针对多处理器操作而设计的处理器中的一种特殊指令,用于管理对共享数据的并发访问;

CAS 是一种无锁的非阻塞算法的实现;

CAS 包含了三个操作数:

需要读写的内存值: V
进行比较的预估值: A

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值