首页 > 其他分享 >JUC学习笔记

JUC学习笔记

时间:2024-11-13 20:47:19浏览次数:3  
标签:JUC Thread lock InterruptedException 笔记 学习 num 线程 new

文章目录


本篇博客是之前学习JUC时记录的内容,对于并发编程知识只是浅浅谈及,并不深入。也算是给自己开新坑。建一个JUC的专栏,后续学习有地方记录。

并发:多个线程操作同一个资源类,把资源丢入线程。

  • 传统Synchronized,本质是锁加队列。
    在这里插入图片描述
  • lock:是一个接口。

lock的实现类有:

  1. ReentrantLock(可重入锁)
  2. ReentrantReadWriteLock.ReadLock(可重入读写锁中的读锁)
  3. ReentrantReadWriteLock.WriteLock(可重入读写锁中的写锁)

ReentrantLock默认实现是非公平锁,可以创建时指定为公平锁。

    public ReentrantLock() {
        sync = new NonfairSync();
    }

    /**
     * Creates an instance of {@code ReentrantLock} with the
     * given fairness policy.
     *
     * @param fair {@code true} if this lock should use a fair ordering policy
     */
    public ReentrantLock(boolean fair) {
        sync = fair ? new FairSync() : new NonfairSync();
    }

公平锁:按照先来后到的顺序依次执行,有一定的次序。
非公平锁:与公平锁相反,不按照一定的次序执行。

synchronized与lock的区别:

  1. Synchronized 是Java的关键字,Lock是Java的接口,对应有接口的实现类ReentrantLock、ReentrantReadWriteLock。
  2. Synchronized 不可以判断获取锁的状态,Lock可以判断获取锁的状态。
  3. Synchronized 会自动释放锁,Lock需要手动释放锁,不释放则死锁。
  4. Synchronized 线程1(获得锁,阻塞),线程2(等待,死等),Lock锁不一定会等待下去,基于内部的tryLock()函数实现。
  5. Synchronized 可重入锁,不可以中断的,非公平的,Lock可重入锁,可以判断锁,非公平(可以自己设置)
  6. Synchronized 适合锁少量的代码同步问题,Lock适合锁大量的同步代码。

生产者消费者问题

生产者和消费者的Synchronized版:
基本步骤:判断等待、执行业务、唤醒(通知)


public class Demo1 {
    public static void main(String[] args) {
        // 创建资源
        Resources resources = new Resources();
        // 执行并发,创建线程,将资源放置到线程中
        new Thread(()->{
            for (int i = 0; i < 10; i++) {
                try {
                    resources.increment();
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
            }
        },"Producer").start();
        new Thread(()->{
            for (int i = 0; i < 10; i++) {
                try {
                    resources.decrement();
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
            }
        },"Consumer").start();
    }
}

class Resources{
    private int num = 0;
    public synchronized void increment() throws InterruptedException {
        // 判断等待
        while(num != 0){
            this.wait();
        }
        // 执行业务
        ++num;
        System.out.println(Thread.currentThread().getName() + "---" + num);
        // 通知其他线程
        this.notifyAll();
    }
    public synchronized void decrement() throws InterruptedException {
        // 判断等待
        while(num == 0){
            this.wait();
        }
        // 执行业务
        --num;
        System.out.println(Thread.currentThread().getName() + "---" + num);
        // 通知其他线程
        this.notifyAll();
    }
}

上面判断条件能否改为if,不能:

if只会判断一次,随后释放锁,另一个线程获得锁后进行if判断,同时处于等待,两个线程均处于等待状态,随后其他线程执行唤醒操作,进入if里面wait的两个线程都执行后面的业务逻辑。if只在进入的时候判断一次,唤醒后if里面存在多个线程此时也不会判断。
if只判断一次,在等待的时候另一个线程修改了数据,那么wait后就不会进行If判断。
使用while判断当多个生产者进入循环并执行等待操作,随后其他线程执行唤醒操作后,生产者仍然会再次进行判断num!=0进而决定是否执行后续业务代码,防止其他生产者更改num后当前生产者仍然执行++num操作。

生产者和消费者的Lock版:

public class Demo2 {
    public static void main(String[] args) {
        // 创建资源
        Resources2 resources2 = new Resources2();
        // 执行并发,创建线程,将资源放置到线程中
        new Thread(() -> {
            for (int i = 0; i < 10; i++) {
                try {
                    resources2.increment();
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
            }
        }, "Producer1").start();
        new Thread(() -> {
            for (int i = 0; i < 10; i++) {
                try {
                    resources2.increment();
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
            }
        }, "Producer2").start();
        new Thread(() -> {
            for (int i = 0; i < 10; i++) {
                try {
                    resources2.increment();
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
            }
        }, "Producer3").start();
        new Thread(() -> {
            for (int i = 0; i < 100; i++) {
                try {
                    resources2.decrement();
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
            }
        }, "Consumer").start();
    }
}

class Resources2{
    private int num = 0;
    // 创建锁
    Lock lock = new ReentrantLock();
    Condition condition = lock.newCondition();
    public void increment() throws InterruptedException {
        // 加锁
        lock.lock();
        try {
            // 判断等待
            while(num != 0){
                condition.await();
            }
            // 执行业务
            ++num;
            System.out.println(Thread.currentThread().getName() + "---" + num);
            // 通知其他线程
            condition.signalAll();
        } catch (InterruptedException e) {
            e.printStackTrace();
        } finally {
            // 解锁
            lock.unlock();
        }
    }
    public void decrement() throws InterruptedException {
        // 加锁
        lock.lock();
        try {
            // 判断等待
            while(num == 0){
                condition.await();
            }
            // 执行业务
            --num;
            System.out.println(Thread.currentThread().getName() + "---" + num);
            // 通知其他线程
            condition.signalAll();
        } catch (InterruptedException e) {
            e.printStackTrace();
        } finally {
            // 解锁
            lock.unlock();
        }
    }
}

使用condition实现的精准唤醒和通知线程:

package condition;

import java.util.concurrent.locks.Condition;
import java.util.concurrent.locks.Lock;
import java.util.concurrent.locks.ReentrantLock;

/**
 * Condition的精准唤醒
 */
public class Demo1 {
    public static void main(String[] args) {
        // 创建资源
        Resource3 resource3 = new Resource3();
        // 创建线程,将资源传给线程,执行对应的并发操作
        new Thread(()->{
            for (int i =0; i < 10; i++) {
                try {
                    resource3.printA();
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
            }
        },"A").start();
        new Thread(()->{
            for (int i =0; i < 10; i++) {
                try {
                    resource3.printB();
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
            }
        },"B").start();
        new Thread(()->{
            for (int i =0; i < 10; i++) {
                try {
                    resource3.printC();
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
            }
        },"C").start();
    }
}
class Resource3{
    // 创建锁
    private Lock lock = new ReentrantLock();
    private Condition condition1 = lock.newCondition();
    private Condition condition2 = lock.newCondition();
    private Condition condition3 = lock.newCondition();
    private int num = 1;
    public void printA() throws InterruptedException {
        // 加锁
        lock.lock();
        try {
            while(num != 1){
                condition1.await();
            }
            System.out.println(Thread.currentThread().getName() + ":num=" + num + "AAAAAA");
            num = 2;
            // 唤醒下一个
            condition2.signal();
        } catch (InterruptedException e) {
            e.printStackTrace();
        } finally {
            lock.unlock();
        }
    }
    public void printB() throws InterruptedException {
        // 加锁
        lock.lock();
        try {
            while(num != 2){
                condition2.await();
            }
            System.out.println(Thread.currentThread().getName() + ":num=" + num + "BBBBBB");
            num = 3;
            // 唤醒下一个
            condition3.signal();
        } catch (InterruptedException e) {
            e.printStackTrace();
        } finally {
            lock.unlock();
        }
    }
    public void printC() throws InterruptedException {
        // 加锁
        lock.lock();
        try {
            while(num != 3){
                condition3.await();
            }
            System.out.println(Thread.currentThread().getName() + ":num=" + num + "CCCCCC");
            num = 1;
            // 唤醒下一个
            condition1.signal();
        } catch (InterruptedException e) {
            e.printStackTrace();
        } finally {
            lock.unlock();
        }
    }
}

上面代码的逻辑是使用Condition唤醒对应的线程,但本身存在一个很大的漏洞,只使用一个Condition仍可以实现上面的流程,由于有个控制变量num的存在,唤醒所有线程后也只会执行num值对应的线程。

8锁现象

集合类不安全

Callable

创建线程的三种方式

常用辅助类

CountDownLatch

Cyclibarrier

Samphore

标签:JUC,Thread,lock,InterruptedException,笔记,学习,num,线程,new
From: https://blog.csdn.net/weixin_45863010/article/details/143750959

相关文章

  • 泷羽sec专题课笔记-- Windows--补充Telnet连接Linux
    本笔记为泷羽sec《红队全栈课程》学习笔记,课程请可自行前往B站学习,课程/笔记主要涉及网络安全相关知识、系统以及工具的介绍等,请使用该课程、本笔记以及课程和笔记中提及工具的读者,遵守网络安全相关法律法规,切勿进行违法违规违纪的操作。写在最前面的话,我们为什么要学习......
  • 一、机器学习算法与实践_07支持向量机与集成学习算法笔记
    1支持向量机1.1定义SVM(SupportVectorMachine,即:支持向量机)是一种监督学习算法,主要用于分类问题,但也可用于回归分析(称为支持向量回归,SupportVectorRegression,简称SVR)1.2核心思想最大间隔原则:SVM试图找到一个超平面(在二维空间中是一条直线,在三维空间中是一个平面,在更......
  • Linux基础笔试练习题笔记(1)
    Linux系统中建立一个新文件可以使用的命令为?A.chmodB.moreC.cpD.touch答案解析:chmod命令是控制用户对文件的权限的命令;more命令类似cat,不过会以一页一页的形式显示,更方便使用者逐页阅读;cp(copyfile)命令主要用于复制文件或目录;touch命令用于修改文件或者目录的时间......
  • 关于AT24C02的学习
    一、基本概念二、特性说明三、内存结构四、器件地址五、指令描述/************************************************************************************@filemain.c*@[email protected]*@versionV1*@date2024-11-13*@briefAT2......
  • 网络学习第四篇
    引言: 我们在第三篇的时候出现了错误,我们要就行排错,那么我们要知道一下怎么配置静态路由实现ping通,这样子我们才知道下一跳到底是什么,为什么这样子做。实验目的理解和掌握静态路由的基本概念和配置方法。实现两个不同网络之间的通信。实验环境两台路由器(AR5和AR6)。三台P......
  • NUXT3学习日记(一)
    Nuxt3是一个基于Vue3的现代框架,用于构建服务器端渲染(SSR)和静态生成的应用程序。它提供了一种简化的方式来创建高性能的Vue应用,具有许多强大的功能和优点。以下是Nuxt3的一些主要应用和优点:一、应用场景服务器端渲染(SSR):Nuxt3可以轻松构建服务器端渲染的应用,这样......
  • 前端技术html中对表单元素的学习
    表单元素目录表单元素基本结构常见的表单元素示例form表单元素在HTML中用于收集用户输入的数据,以便将数据发送到服务器进行处理。表单可以包含多种类型的输入元素,如文本字段、密码字段、单选按钮、复选框、下拉选择菜单、提交按钮等。用户填写表单后,通常通过点击提交按钮将......
  • 【深度学习|课程笔记1】何恺明大神在MIT的秋季最新课程!附课程地址
    【深度学习|课程笔记1】何恺明大神在MIT的秋季最新课程!附课程地址【深度学习|课程笔记1】何恺明大神在MIT的秋季最新课程!附课程地址文章目录【深度学习|课程笔记1】何恺明大神在MIT的秋季最新课程!附课程地址课程概览欢迎宝子们点赞、关注、收藏!欢迎宝子们批评指......
  • rust学习八、包和模块
    总体上,也没有什么特别的地方,和其它语言比较起来。我们可以看懂熟悉的字眼:包括、模块、use、公共等等如果是英文,则需要知道crate、pub。本章节对应相关书籍的第七章节.一、一个rust可执行程序的大体结构就本章节而言,尚未接触到一个非常复杂的工程结构,据说有什么工作空间。不......
  • 大数据学习13之Scala基础语法(重点)
    1.简介        Scala是ScalableLanguage的简写,是一门多范式的编程语言。创始人为MartinOdersky马丁·奥德斯基。        Scala这个名字来源于ScalableLanguage(可伸缩的语言),它是一门基于JVM的多范式编程语言,通俗的说:Scala是一种运行在JVM上的......