1.实现多用户同时取钱
单线程run方法中死循环实现的代码(每个客户每次取1000元)
atm1.deposite -= 1000;
System.out.println(getName() + "取出后剩余" + atm1.deposite);
银行系统同时接受两个客户(两个线程)的访问
public class BankSystem {
int deposite=100000;
public static void main(String[]args){
UserThread u1=new UserThread(atm1);
u1.setName("客户1:");
u1.start();
UserThread u2=new UserThread(atm1);
u2.setName("客户2:");
u2.start();
}
}
结果呈现
由结果我们可以看出,取钱结果并不像我们所设想的那样,一个客户取了1000之后还剩99000,每取一次减少1000,依次类推
所以如何解决这种多线程共用数据时的安全问题呢?
2.解决线程安全问题
1.使用同步代码块
格式为:
synchronized(监视器对象){
需要同步的代码
}
例:
synchronized (atm1){
atm1.deposite -= 1000;
System.out.println(getName() + "取出后剩余" + atm1.deposite);
}
达到的效果:
同一时间只有一个线程可以可以执行代码块中的代码,其他线程必须等待当前线程执行完代码块中的任务后,才可以执行代码块中的任务
这里做一个类比:
有一把钥匙和只能容纳一个人的小屋,四个人想进这个小屋(四个线程想执行任务),当一个人拿着钥匙进了这个小屋之后,其他人只能在屋外等待,当这个人出来之后,谁拿到了钥匙,谁就可以进这个小屋
结果展示:
这正是我们想要的效果
注意事项
监听对象必须是唯一的(就像开门的钥匙只能有一把)
2.使用synchronized修饰的方法
格式为:
public synchronize void 方法名(){
需要同步的代码
}
例:
public synchronized void draw(){
atm1.deposite-=1000;
System.out.println("取出后剩余"+atm1.deposite);
}
注意事项
监听对象是当前类对象
还记得上一种解决线程安全问题需要注意什么吗?
要保证监听对象的唯一性
这里也是一样的
所以在使用这种方法解决线程安全问题时,只能用实现Runnable类来解决
继承Thead类这种方法,在实现多线程时,必须创建多个对象,不能满足监听对象的唯一性