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 方法是类一加载就有了。