数据安全+线程锁+多线程访问临界资源时的数据安全问题
一、线程的生命周期
1、新建状态
1.在程序中用构造方法创建了一个线程对象后,新的线程对象便处于新建状态,此时,它已经有了相应的内存空间和其它资源,但还处于不可运行状态。新建一个线程对象可采用线程构造方法来实现。
2.例如:Thread thread=new Thread();
2、 就绪状态
新建线程对象后,调用该线程的start()方法就可以启动线程。当线程启动时,线程进入就绪状态。此时,线程将进入线程队列排队,等待CPU调用,这表明它已经具备了运行条件。
3、运行状态
当就绪状态的线程被调用并获得处理器资源时,线程就进入了运行状态。此时,自动调用该线程对象的run()方法。run()方法定义了该线程的操作和功能。
4、 阻塞状态
一个正在执行的线程在某些特殊情况下,如被人为挂起,将让出CPU并暂时中止自己的执行,进入阻塞状态。在可执行状态下,如果调用sleep()、wait()等方法,线程都将进入阻塞状态。阻塞时,线程不能进入排队队列,只有当引起阻塞的原因被消除后,线程才可以转入就绪状态。
5、死亡状态
线程调用stop()方法时或run()方法执行结束后,线程即处于死亡状态。处于死亡状态的线程不具有继续运行的能力。
二、数据安全
需求 铁道部发布了一个售票任务,要求销售1000张票,由3个窗口进行销售
请编写多线程程序来模拟这个效果
窗口001正在销售第1张票
窗口001正在销售第2张票
窗口002正在销售第3张票
。。。
窗口002正在销售第1000张票
使用线程类的方式去解决需求
问题一:三个窗口各买1000张票(一共卖了3000张票)
出现原因:创建三个线程,每个线程都独享一份成员变量(maxTicket和curTicket)
解决方案:三个线程共享属性(maxTicket和curTicket) — 静态属性
问题二:有些票没有卖到,有些票卖了重票
出现原因:当前线程让curTicket自增后还没来得及输出,就退出CPU资源,其他线程又抢到资源了
解决方案:当前线程让curTicket自增后必须输出完才能退出CPU资源
问题三:多卖了票
出现原因:curTicket等于999时,三个线程互相抢资源,都进入到循环中
解决方案:在锁里再判断一次
何为线程安全问题?
多线程去操作同一个资源,就容易出现线程安全问题(脏数据)
线程安全的方式 – 加锁
经验:多个线程想互斥住,必须使用同一把锁对象
synchronized
同步代码块:
synchronized(锁对象){//自动上锁
…想要同步的代码…
}//自动解锁
public class Test01 {
public static void main(String[] args) {
MyThread t1 = new MyThread("窗口001");
MyThread t2 = new MyThread("窗口002");
MyThread t3 = new MyThread("窗口003");
t1.start();
t2.start();
t3.start();
}
}
public class MyThread extends Thread{
private static int maxTicket = 1000;//最大票数
private static int curTicket = 0;//当前卖出的票数
private static Object obj = new Object();
public MyThread(String name) {
super(name);
}
@Override
public void run() {
while (curTicket < maxTicket) {
synchronized (obj) {
if (curTicket < maxTicket) {
curTicket++;
System.out.println(Thread .currentThread().getName() + "正在销售第" + curTicket + "张票");
}
if (curTicket == maxTicket) {
System.out.println(Thread.currentThread().getName()+ "票已售完");
}
}
}
}
}
三、线程锁
1.创建线程的方式 – 线程类
同步方法:
成员的同步方法: 锁对象 -- this
public synchronized void method(){//自动上锁
...想要同步的代码...
}//自动解锁
静态的同步方法:锁对象 -- 该类的class对象
public static synchronized void method(){//自动上锁
...想要同步的代码...
}//自动解锁
public class Test01 {
public static void main(String[] args) {
MyThread t1 = new MyThread("窗口001");
MyThread t2 = new MyThread("窗口002");
MyThread t3 = new MyThread("窗口003");
t1.start();
t2.start();
t3.start();
}
}
public class MyThread extends Thread{
private static int maxTicket = 1000;//最大票数
private static int curTicket = 0;//当前卖出的票数
public MyThread(String name) {
super(name);
}
@Override
public void run() {
while (curTicket < maxTicket) {
method();
}
}
public static synchronized void method() {
if(curTicket < maxTicket){
curTicket++;
System.out.println(Thread.currentThread().getName() + "正在销售第" + curTicket + "张票");
}
if