数据安全+线程锁+多线程访问临界资源时的数据安全问题

数据安全+线程锁+多线程访问临界资源时的数据安全问题

一、线程的生命周期

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
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

雨霖先森

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值