目录
1_7_3_01_线程安全问题的概述
多线程访问共享数据会产生线程安全问题
1_7_3_02_线程安全问题的代码实现
E:\0-2019黑马\1.Java基础笔记\17.【线程、同步】-笔记\code\06_Thread\src\com\itheima\demo06\ThreadSafe
运行此程序之后,发现有以下的问题:
重复的票
不存在的票
1_7_3_03_线程安全问题产生的原理
相当于没有关上门,同时访问一个共享资源。
1_7_3_04_解决线程安全问题_同步代码块
E:\0-2019黑马\1.Java基础笔记\17.【线程、同步】-笔记\code\06_Thread\src\com\itheima\demo07\Synchronized
卖票案例出现了线程安全问题
卖出了不存在的票和重复的票
解决线程安全问题的一种方案:使用同步代码块
格式:
synchronized(锁对象){ 对象 object
可能会出现线程安全问题的代码(访问了共享数据的代码)
}
注意:
1.通过代码块中的锁对象,可以使用任意的对象
2.但是必须保证多个线程使用的锁对象是同一个
3.锁对象作用:
把同步代码块锁住,只让一个线程在同步代码块中执行
主要代码;使用同步代码块解决安全问题。
synchronized(锁对象){
可能会出现线程安全问题的代码(访问了共享数据的代码)
}
把同步代码块锁住,只让一个线程在同步代码块中执行
//部分代码
//创建一个锁对象
Object obj = new Object();
//设置线程任务:卖票
@Override
public void run() {
//使用死循环,让卖票操作重复执行
while(true){
//同步代码块
synchronized (obj){
//先判断票是否存在
if(ticket>0){
//提高安全问题出现的概率,让程序睡眠;有利于安全问题的出现!哈哈哈!
try {
Thread.sleep(10);
} catch (InterruptedException e) {
e.printStackTrace();
}
//票存在,卖票 ticket--
System.out.println(Thread.currentThread().getName()+"-->正在卖第"+ticket+"张票");
ticket--;
}
}
}
}
}
1_7_3_05_同步技术的原理
程序的效率会降低,但是会保证程序安全。
使用了一个同步锁对象,检查synchronized代码块是否有锁对象,把obj看成开门进入的钥匙
1_7_3_06_解决线程安全问题_同步方法
是因为编程软件太占内存了嘛!
1_7_3_06_解决线程安全问题_ 同步方法
1_7_3_07_静态同步方法
1_7_3_06_解决线程安全问题_ 同步方法 和 1_7_3_07_静态同步方法 没听懂
1_7_3_08_解决线程安全问题_Lock锁
E:\0-2019黑马\1.Java基础笔记\17.【线程、同步】-笔记\code\06_Thread\src\com\itheima\demo09\Lock
卖票案例出现了线程安全问题
卖出了不存在的票和重复的票解决线程安全问题的三种方案:使用Lock锁
java.util.concurrent.locks.Lock接口
Lock 实现提供了比使用 synchronized 方法和语句可获得的更广泛的锁定操作。
Lock接口中的方法:
void lock()获取锁。
void unlock() 释放锁。
java.util.concurrent.locks.ReentrantLock implements Lock接口
使用步骤:
1.在成员位置创建一个ReentrantLock对象
2.在可能会出现安全问题的代码前调用Lock接口中的方法lock获取锁
3.在可能会出现安全问题的代码后调用Lock接口中的方法unlock释放锁
下面代码中使用了finally语句,finally语句一定执行
import java.util.concurrent.locks.Lock;
import java.util.concurrent.locks.ReentrantLock;
public class RunnableImpl implements Runnable{
//定义一个多个线程共享的票源
private int ticket = 100;
//1.在成员位置创建一个ReentrantLock对象
Lock l = new ReentrantLock();
//设置线程任务:卖票
@Override
public void run() {
//使用死循环,让卖票操作重复执行
while(true){
//2.在可能会出现安全问题的代码前调用Lock接口中的方法lock获取锁
l.lock();
//先判断票是否存在
if(ticket>0){
//提高安全问题出现的概率,让程序睡眠
try {
Thread.sleep(10);
//票存在,卖票 ticket--
System.out.println(Thread.currentThread().getName()+"-->正在卖第"+ticket+"张票");
ticket--;
} catch (InterruptedException e) {
e.printStackTrace();
}finally {
//3.在可能会出现安全问题的代码后调用Lock接口中的方法unlock释放锁
l.unlock();//无论程序是否异常,都会把锁释放
}
}
}
}
}