1、线程出现安全的问题的原因
在多线程并发运行,并且实现资源共享的时候,有可能出现线程安全的问题,几个线程拿到共享资源的值都是相同的一个最新值,这个值的前面的数据一个都没有拿到,线程拿到的值都是一样的,或者有一些值没有被拿到,这是多线程并发共享资源的时候出现的线程安全。
如:模拟一个车票购买的情况说明线程的安全问题,首先车票的数量要共享给线程使用,抢到票要进行处理或者返回到界面显示、或者进行支付,这些操作都需要时间进行,这里就有线程睡眠来代替,每抢到一张票,都进行100毫秒睡眠。
子线程代码:
package com.ticket;
public class ticketThread implements Runnable{
private int mTicket=10;
private int mCount=0;
@Override
public void run() {
while (mTicket>0){
mTicket--;
mCount++;
try {
Thread.sleep(100);
} catch (InterruptedException e) {
e.printStackTrace();
}
System.out.println(Thread.currentThread().getName()+"购买:"+mCount+"号票,还剩"+mTicket+"张票");
}
}
}
主线程代码:
package com.ticket;
public class Mian {
public static void main(String[] args) {
Runnable runnable=new ticketThread();
Thread t1=new Thread(runnable,"小张");
Thread t2=new Thread(runnable,"小王");
Thread t3=new Thread(runnable,"小赵");
t1.start();
t2.start();
t3.start();
}
}
结果:出现了线程问题:多人购买到了同一张票,有的票没有被购买过
2、解决线程安全的问题
线程安全是因为线程同时操作共同的资源,然后因为一些操作需要时间来处理,并且返回的结果都是最新的,其他的一些结果在操作的时候被忽略掉了,这时我们就需要确保这一些代码可以单独运行,等到结果出来后再进行其他线程的操作,并且也确保这个线程单独运行,直到每一个线程都单独运行完。能够帮我们实现这一个功能的方法是为这些代码块添加一个同步锁,确保这一个线程运行完后,才可以执行下一个线程。
1)同步锁(synchronized)介绍
添加了同步锁的代码块或者方法,这些代码块或者方法在线程运行到这里后,直接给这些代码块或方法上了一把锁,这把锁确定其他线程不能访问,直到运行完者代码块或方法后,这把锁就会打开,其他的线程就可以访问这个代码块或者方法了。
2)同步锁使用
同步锁在代码块上使用:
synchronized(this){
代码块
}
同步锁在方法上使用:
放在public修饰符后面
public synchronized void add(){
}
放在public修饰符前面
synchronized public void add(){
}
2)使用同步锁解决线程安全问题
子线程代码:
package com.ticket;
public class ticketThread implements Runnable{
private int mTicket=10;
private int mCount=0;
@Override
public void run() {
while (mTicket>0){
synchronized (this){
if(mTicket>0){
mTicket--;
mCount++;
try {
Thread.sleep(100);
} catch (InterruptedException e) {
e.printStackTrace();
}
System.out.println(Thread.currentThread().getName()+"购买:"+mCount+"号票,还剩"+mTicket+"张票");
}
}
}
}
}
主线程代码:
package com.ticket;
public class Mian {
public static void main(String[] args) {
Runnable runnable=new ticketThread();
Thread t1=new Thread(runnable,"小张");
Thread t2=new Thread(runnable,"小王");
Thread t3=new Thread(runnable,"小赵");
t1.start();
t2.start();
t3.start();
}
}
效果: