线程安全问题的概述
线程安全问题的代码实现
package demo03;
/*
模拟卖票案例
创建三个线程,同时开启,对共享的票进行出售
*/
public class Demo01Ticket {
public static void main(String[] args) {
//创建Runnable接口的实现类对象
RunnableImpl run = new RunnableImpl();
//创建Thread类对象,构造方法中传递Runnable接口中的实现类对象
Thread t0 = new Thread(run);
Thread t1 = new Thread(run);
Thread t2 = new Thread(run);
//调用start方法开启多线程
t0.start();
t1.start();
t2.start();
}
}
package demo03;
/*
实现买票案例
*/
public class RunnableImpl implements Runnable{
//定义一个多个线程共享的资源
private int ticket = 100;
//设置线程任务:卖票
@Override
public void run() {
//使用死循环,让卖票操作重复执行
while(true) {
//先判断是否存在
if (ticket>0) {
//提高安全问题出现的概率,让程序睡眠
try {
Thread.sleep(10);
} catch (InterruptedException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
System.out.println(Thread.currentThread().getName()+"-->正在卖第"+ticket+"张票");
ticket--;
}
}
}
}
线程安全问题产生的原理
解决线程安全问题 同步代码块
package demo04;
/*
模拟卖票案例
创建三个线程,同时开启,对共享的票进行出售
*/
public class Demo01Ticket {
public static void main(String[] args) {
//创建Runnable接口的实现类对象
RunnableImpl run = new RunnableImpl();
//创建Thread类对象,构造方法中传递Runnable接口中的实现类对象
Thread t0 = new Thread(run);
Thread t1 = new Thread(run);
Thread t2 = new Thread(run);
//调用start方法开启多线程
t0.start();
t1.start();
t2.start();
}
}
package demo04;
/*
卖票案例出现了线程安全问题
卖出了不存在的票和重复的票
解决线程安全问题的一种解决方案:使用同步代码块
格式:
synchronized(锁对象){
可能会出现线程安全问题的代码(访问了共享数据的代码)
}
注意:
1、通过代码中的锁对象,可以使用任意的对象
2、但是必须保证多个线程使用的锁对象是同一个
3、锁对象作用:
把同步代码块锁住,只让一个线程在同步代码块中执行
*/
public class RunnableImpl implements Runnable{
//定义一个多个线程共享的资源
private int ticket = 100;
//创建一个锁对象
Object obj = new Object();
//设置线程任务:卖票
@Override
public void run() {
//使用死循环,让卖票操作重复执行
while(true) {
//同步代码块
synchronized (obj) {
//先判断是否存在
if (ticket>0) {
//提高安全问题出现的概率,让程序睡眠
try {
Thread.sleep(10);
} catch (InterruptedException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
//票存在,卖票ticket--
System.out.println(Thread.currentThread().getName()+"-->正在卖第"+ticket+"张票");
ticket--;
}
}
}
}
}
同步技术的原理
解决线程安全问题 同步方法
package demo05;
/*
模拟卖票案例
创建三个线程,同时开启,对共享的票进行出售
*/
public class Demo01Ticket {
public static void main(String[] args) {
//创建Runnable接口的实现类对象
RunnableImpl run = new RunnableImpl();
System.out.println("run:"+run);
//创建Thread类对象,构造方法中传递Runnable接口中的实现类对象
Thread t0 = new Thread(run);
Thread t1 = new Thread(run);
Thread t2 = new Thread(run);
//调用start方法开启多线程
t0.start();
t1.start();
t2.start();
}
}
package demo05;
/*
卖票案例出现了线程安全问题
卖出了不存在的票和重复的票
解决线程安全问题的第二种解决方案:使用同步方法
使用步骤:
1、把访问的共享数据的代码抽取出来,放到一个方法中
2、在方法上添加synchroized修饰符
格式:定义方法的格式
修饰符:synchronized 返回值类型 方法名(参数列表){
可能会出现的线程安全问题的代码(访问了共享数据的代码)
}
*/
public class RunnableImpl implements Runnable{
//定义一个多个线程共享的资源
private int ticket = 100;
//创建一个锁对象
Object obj = new Object();
//设置线程任务:卖票
@Override
public void run() {
System.out.println(this);
//使用死循环,让卖票操作重复执行
while(true) {
payTicket();
}
}
/*
定义一个同步方法
同步方法也会把方法内部的代码锁住
只让一个线程执行
同步方法的锁对象是谁?
就是实现类对象 new RunnableImpl()
也就是this
*/
public /*synchronized*/ void payTicket() {
//同步代码块
synchronized (this) {
//先判断是否存在
if (ticket>0) {
//提高安全问题出现的概率,让程序睡眠
try {
Thread.sleep(10);
} catch (InterruptedException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
//票存在,卖票ticket--
System.out.println(Thread.currentThread().getName()+"-->正在卖第"+ticket+"张票");
ticket--;
}
}
}
}
静态同步方法
package demo05;
/*
模拟卖票案例
创建三个线程,同时开启,对共享的票进行出售
*/
public class Demo01Ticket {
public static void main(String[] args) {
//创建Runnable接口的实现类对象
RunnableImpl run = new RunnableImpl();
System.out.println("run:"+run);
//创建Thread类对象,构造方法中传递Runnable接口中的实现类对象
Thread t0 = new Thread(run);
Thread t1 = new Thread(run);
Thread t2 = new Thread(run);
//调用start方法开启多线程
t0.start();
t1.start();
t2.start();
}
}
package demo05;
/*
卖票案例出现了线程安全问题
卖出了不存在的票和重复的票
解决线程安全问题的第二种解决方案:使用同步方法
使用步骤:
1、把访问的共享数据的代码抽取出来,放到一个方法中
2、在方法上添加synchroized修饰符
格式:定义方法的格式
修饰符:synchronized 返回值类型 方法名(参数列表){
可能会出现的线程安全问题的代码(访问了共享数据的代码)
}
*/
public class RunnableImpl implements Runnable{
//定义一个多个线程共享的资源
private static int ticket = 100;
//创建一个锁对象
Object obj = new Object();
//设置线程任务:卖票
@Override
public void run() {
System.out.println(this);
//使用死循环,让卖票操作重复执行
while(true) {
payTicketStatic();
}
}
/*
静态的同步方法
锁对象是谁?
不能是this
this是创建对象之后产生的,静态方法优先于对象
静态方法的锁对象是本类的class属性-->class文件对象(反射)
*/
public static synchronized void payTicketStatic() {
synchronized(RunnableImpl.class) {
if (ticket>0) {
//提高安全问题出现的概率,让程序睡眠
try {
Thread.sleep(10);
} catch (InterruptedException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
//票存在,卖票ticket--
System.out.println(Thread.currentThread().getName()+"-->正在卖第"+ticket+"张票");
ticket--;
}
}
}
}
解决线程安全问题 Lock锁
package demo05;
/*
模拟卖票案例
创建三个线程,同时开启,对共享的票进行出售
*/
public class Demo01Ticket {
public static void main(String[] args) {
//创建Runnable接口的实现类对象
RunnableImpl run = new RunnableImpl();
System.out.println("run:"+run);
//创建Thread类对象,构造方法中传递Runnable接口中的实现类对象
Thread t0 = new Thread(run);
Thread t1 = new Thread(run);
Thread t2 = new Thread(run);
//调用start方法开启多线程
t0.start();
t1.start();
t2.start();
}
}
package demo05;
import java.util.concurrent.locks.Lock;
import java.util.concurrent.locks.ReentrantLock;
/*
卖票案例出现了线程安全问题
卖出了不存在的票和重复的票
解决线程安全问题的第三种解决方案:使用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释放锁
*/
public class RunnableImpl implements Runnable{
//定义一个多个线程共享的资源
private int ticket = 100;
//1、在成员位置创建一个Reentrantlock对象
Lock l = new ReentrantLock();
//设置线程任务:卖票
@Override
public void run() {
System.out.println(this);
//使用死循环,让卖票操作重复执行
while(true) {
//2、在可能会出现安全问题的代码前调用Lock接口中的方法lock获取锁
l.lock();
//先判断票是否存在
if (ticket>0) {
//提高安全问题出现的概率,让程序睡眠
try {
Thread.sleep(10);
} catch (InterruptedException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}finally {
//3、在可能会出现安全问题的代码后调用lock接口中的方法unlock释放锁
l.unlock();//无论程序是否异常,都会把锁给释放掉
}
//票存在,卖票ticket--
System.out.println(Thread.currentThread().getName()+"-->正在卖第"+ticket+"张票");
ticket--;
}
}
}
}