JUC学习笔记及实操(一)

JUC
1、什么是JUC

Java.Util 工具包,用来处理并发问题

Java.Util.Concurrent

2、进程和线程

进程:一个程序,程序的集合

线程:开了一个进程Typora,码字、自动保存(线程负责的)

Java默认有2个线程:main、GC

对于Java而言:Thread、Runnable、Callable

Java 真的可以开启线程吗?

开不了。它需要调用本地方法,本地方法的底层是C++,而Java是无法直接操作硬件的

并发、并行

并发:多条线程操作同一资源

  • CPU单核,模拟出来多条线程,快速交替

并行:多个线程可以同时执行

  • CPU多核,多个线程可以同时执行;线程池

并发编程的本质:充分利用CPU的资源

线程有几个状态

  • 新生 NEW
  • 运行 RUNNABLE
  • 阻塞 BLOCKED
  • 等待 WAITING
  • 超时等待 TIMED_WATTING
  • 终止 TERMINATED

wait/sleep区别

1、来自不同的类

wait -> Object

sleep -> Thread

2、关于锁的释放

wait会释放锁,sleep睡觉了,抱着锁睡觉,不会释放

3、使用的范围不同

wait必须在同步代码块中

sleep可以在任何地方睡

3、Lock锁(重点)

传统做法:

synchronized:本质就是队列

lock接口

可重入锁:ReentrantLock

公平锁:十分公平,可以先来后到

非公平锁:十分不公平,可以插队(默认)

private final ReentrantLock lock = new ReentrantLock();

public void run () {
	lock.lock();	// 加锁
	try {
       // 业务代码 
    } catch (Exception e) {
        e.printStackTrace();
    } finally {
        lock.unlock();	// 解锁
    }
}

Synchronized 和 Lock 的区别

1、Synchronized 内置Java关键字,Lock是一个Java类

2、Synchronized 无法判断获取锁的状态,Lock可以判断是否获得到了锁

3、Synchronized 会自动释放锁,Lock必须手动释放锁,如果不释放锁,死锁

4、Synchronized 线程1(获得锁,阻塞)、线程2(等待,傻傻的等);Lock锁就不一定会等待下去

5、Synchronized 可重入锁,不可以中断的,非公平;Lock 可重入锁,可以判断锁,非公平(可以自己设置)

6、Synchronized 适合锁少量的代码同步问题,Lock适合锁大量的同步代码。

4、生产者和消费者问题

面试:单例模式、排序算法、生产者和消费者、死锁

生产者 消费者问题 Synchronized 版

步骤:判断等待、业务、通知

// 判断等待,业务,通知
class Data{	// 数字 资源类
	private int number = 0;
	//+1
	public synchronized void increment() throws InterruptedException {
        if (number!=0){ //0
            // 等待
            this .wait();
        }
		number++;
        System.out.printIn(Thread.currentThread().getName()+"=>"+number);
        // 通知其他线程,我+1完毕了
        this .notifyA11();
    }
	//-1
	public synchronized void decrement() throws InterruptedException {
        if (number==0){ // 1
            // 等待
            this .wait();
        }
        number--;
        System.out.printIn(Thread.currentThread().getName()+"=>"+number);
        // 通知其他线程,我-1完毕了
		this .notifyA11() ;
    }
}

A、B、C、D 四个线程?虚假唤醒

为了防止出现虚假唤醒,要把上面代码的 if 判断语句改为 while 循环语句,等待应该总是出现在循环中。

JUC版的生产者和消费者问题

在这里插入图片描述

通过Lock找到Condition

Lock lock = new ReentrantLock();
Condition condition = lock.newCondition();
condition.await();
condition.signalAll();

步骤:

lock.lock();
try{
	//业务代码
	while(判断条件){
		condition.await();
	}
    condition.signalAll();
} catch (Exception e){
    e.printStackTrace();
} finally {
    lock.unlock();
}

Condition 精准的通知和唤醒线程

class Data {
   private int num = 1;
   private Lock lock = new ReentrantLock();
   private Condition condition1 = lock.newCondition();
   private Condition condition2 = lock.newCondition();
   private Condition condition3 = lock.newCondition();

   public void printA() {
       lock.lock();
       try {
           while (num != 1) {
               condition1.await();
           }
           System.out.println(Thread.currentThread().getName()+"=> AAAAAA");
           num = 2;
           condition2.signal();
       } catch (Exception e) {
           e.printStackTrace();
       } finally {
           lock.unlock();
       }

   }

   public void printB() {
       lock.lock();
       try {
           while (num != 2) {
               condition2.await();
           }
           System.out.println(Thread.currentThread().getName()+"=> BBBBBB");
           num = 3;
           condition3.signal();
       } catch (Exception e) {
           e.printStackTrace();
       } finally {
           lock.unlock();
       }

   }

   public void printC() {
       lock.lock();
       try {
           while (num != 3) {
               condition3.await();
           }
           System.out.println(Thread.currentThread().getName()+"=> CCCCCC");
           num = 1;
           condition1.signal();
       } catch (Exception e) {
           e.printStackTrace();
       } finally {
           lock.unlock();
       }
   }
}
5、8锁现象

关于锁的8个问题

如何判断锁的是谁?永远的知道什么锁,锁到底锁的是谁

1、标准情况下,两个线程先打印 发短信还是 打电话? 1/发短信 2/打电话

2、sendSms延迟4秒,两个线程先打印 发短信还是 打电话? 1/发短信 2/打电话

/**
* 8锁,就是关于锁的8个问题
* 1、标准情况下,两个线程先打印 发短信还是 打电话? 1/发短信 2/打电话
* 2、sendSms延迟4秒,两个线程先打印 发短信还是 打电话? 1/发短信 2/打电话
*/
public class Test1 {
    public static void main(String[] args) {
        Phone phone = new Phone();
        //锁的存在
        new Thread(()->{
            phone.sendSms();
        },"A").start();
        // 捕获
        try {
            TimeUnit.SECONDS.sleep(1);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
        new Thread(()->{
            phone.call();
        },"B").start();
    }
}
class Phone{
    // synchronized 锁的对象是方法的调用者!、
    // 两个方法用的是同一个锁,谁先拿到谁执行!
    public synchronized void sendSms(){
        try {
            TimeUnit.SECONDS.sleep(4);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
        System.out.println("发短信");
    }
    public synchronized void call(){
        System.out.println("打电话");
    }
}

3、增加了一个普通方法后!先执行发短信还是Hello? 普通方法

4、两个对象,两个同步方法, 发短信还是 打电话? // 打电话

/**
* 3、 增加了一个普通方法后!先执行发短信还是Hello? 普通方法
* 4、 两个对象,两个同步方法, 发短信还是 打电话? // 打电话
*/
public class Test2 {
    public static void main(String[] args) {
        // 两个对象,两个调用者,两把锁!
        Phone2 phone1 = new Phone2();
        Phone2 phone2 = new Phone2();
        //锁的存在
        new Thread(()->{
            phone1.sendSms();
        },"A").start();
        // 捕获
        try {
            TimeUnit.SECONDS.sleep(1);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
        new Thread(()->{
            phone2.call();
        },"B").start();
    }
}
class Phone2{
    // synchronized 锁的对象是方法的调用者!
    public synchronized void sendSms(){
        try {
            TimeUnit.SECONDS.sleep(4);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
        System.out.println("发短信");
    }
    public synchronized void call(){
        System.out.println("打电话");
    }
    // 这里没有锁!不是同步方法,不受锁的影响
    public void hello(){
        System.out.println("hello");
    }
}

5、增加两个静态的同步方法,只有一个对象,先打印 发短信?打电话?

6、两个对象!增加两个静态的同步方法, 先打印 发短信?打电话?

/**
* 5、增加两个静态的同步方法,只有一个对象,先打印 发短信?打电话?
* 6、两个对象!增加两个静态的同步方法, 先打印 发短信?打电话?
*/
public class Test3 {
    public static void main(String[] args) {
        // 两个对象的Class类模板只有一个,static,锁的是Class
        Phone3 phone1 = new Phone3();
        Phone3 phone2 = new Phone3();
        //锁的存在
        new Thread(()->{
            phone1.sendSms();
        },"A").start();
        // 捕获
        try {
            TimeUnit.SECONDS.sleep(1);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
        new Thread(()->{
            phone2.call();
        },"B").start();
    }
}
// Phone3唯一的一个 Class 对象
class Phone3{
    // synchronized 锁的对象是方法的调用者!
    // static 静态方法
    // 类一加载就有了!锁的是Class
    public static synchronized void sendSms(){
        try {
            TimeUnit.SECONDS.sleep(4);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
        System.out.println("发短信");
    }
    public static synchronized void call(){
        System.out.println("打电话");
    }
}

7、1个静态的同步方法,1个普通的同步方法 ,一个对象,先打印 发短信?打电话?

8、1个静态的同步方法,1个普通的同步方法 ,两个对象,先打印 发短信?打电话?

/**
* 7、1个静态的同步方法,1个普通的同步方法 ,一个对象,先打印 发短信?打电话?
* 8、1个静态的同步方法,1个普通的同步方法 ,两个对象,先打印 发短信?打电话?
*/
public class Test4 {
    public static void main(String[] args) {
        // 两个对象的Class类模板只有一个,static,锁的是Class
        Phone4 phone1 = new Phone4();
        Phone4 phone2 = new Phone4();
        //锁的存在
        new Thread(()->{
            phone1.sendSms();
        },"A").start();
        // 捕获
        try {
            TimeUnit.SECONDS.sleep(1);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
        new Thread(()->{
            phone2.call();
        },"B").start();
    }
}
// Phone3唯一的一个 Class 对象
class Phone4{
    // 静态的同步方法 锁的是 Class 类模板
    public static synchronized void sendSms(){
        try {
            TimeUnit.SECONDS.sleep(4);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
        System.out.println("发短信");
    }
    // 普通的同步方法 锁的调用者
    public synchronized void call(){
        System.out.println("打电话");
    }
}

小结

深刻理解锁

  • synchronized 锁的对象是方法的调用者,具体的对象

​ 两个方法用的是同一个锁,谁先拿到谁先执行

  • 如果再加上 static 修饰,则锁的是 Class 模板;static 方法是类一加载就有了。
  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值