在这里插入图片描述
5)
package com.bjpowernode.demo03;
/**
* 通过线程模拟多人同时从同一帐户中取钱
* @author Administrator
*
*/
public class Test01 {
public static void main(String[] args) {
//先开户
MyBankAccount bankAccount = new MyBankAccount();
//创建三个线程模拟三个人,从同一帐户取钱
PersonThread p1 = new PersonThread(bankAccount);
PersonThread p2 = new PersonThread(bankAccount);
PersonThread p3 = new PersonThread(bankAccount);
p1.setName("ganggang");
p2.setName("yuanyuan");
p3.setName("bingbing");
p1.start();
p2.start();
p3.start();
/*
* 某一次的执行结果为:
ganggang取钱前,余额为:10000
bingbing取钱前,余额为:10000
bingbing取了1000元后,余额为:8000
yuanyuan取钱前,余额为:10000
yuanyuan取了1000元后,余额为:7000
ganggang取了1000元后,余额为:9000
现象:
yuanyuan线程取钱 前/后的数额不正确,这种线程称为线程安全问题, 也叫线程不安全问题
分析原因:
一个线程在执行取钱操作期间,被其他的线程插入进来进行了另外的操作
解决思路:
在线程执行这个取钱操作时,不允许其他线程加入
*/
}
}
package com.bjpowernode.demo03;
/**
* 银行帐户
* @author Administrator
*
*/
public class MyBankAccount {
int balance = 10000; //余额
//取钱的操作,约定一次取1000
public void withdraw() {
System.out.println( Thread.currentThread().getName() + "取钱前,余额为:" + balance);
balance -= 1000;
System.out.println( Thread.currentThread().getName() + "取了1000元后,余额为:" + balance);
}
}
package com.bjpowernode.demo03;
/**
* 创建线程类,模拟人取钱
* @author Administrator
*
*/
public class PersonThread extends Thread {
MyBankAccount myAccount ; //取钱的帐户
//构造方法
public PersonThread(MyBankAccount myAccount) {
super();
this.myAccount = myAccount;
}
@Override
public void run() {
myAccount.withdraw();
}
}
加入同步方法
package com.bjpowernode.demo04;
/**
* 通过线程模拟多人同时从同一帐户中取钱
* @author Administrator
*
*/
public class Test01 {
public static void main(String[] args) {
//先开户
MyBankAccount bankAccount = new MyBankAccount();
//创建三个线程模拟三个人,从同一帐户取钱
PersonThread p1 = new PersonThread(bankAccount);
PersonThread p2 = new PersonThread(bankAccount);
PersonThread p3 = new PersonThread(bankAccount);
p1.setName("ganggang");
p2.setName("yuanyuan");
p3.setName("bingbing");
p1.start();
p2.start();
p3.start();
}
}
package com.bjpowernode.demo04;
/**
* 银行帐户
* @author Administrator
*
*/
public class MyBankAccount {
int balance = 10000; //余额
private static final Object LOCK = new Object(); //定义一个常量, 作为锁对象
//取钱的操作,约定一次取1000
public void withdraw() {
synchronized (LOCK) { //经常使用常量作为锁对象
System.out.println( Thread.currentThread().getName() + "取钱前,余额为:" + balance);
balance -= 1000;
System.out.println( Thread.currentThread().getName() + "取了1000元后,余额为:" + balance);
}
}
/*
* 线程同步语法:
* synchronized( 锁对象 ) {
* 同步代码块
* }
* 工作原理:
* 1)线程想执行同步代码块,必须先获得锁对象
* 2)任意对象都可以作为锁对象, 所有对象都有一个内置锁
* 3) 锁对象在某一时刻最多只能由一个线程持有
* 4)线程持有了锁对象后会一直持有, 直到它执行完同步代码块后再释放
* 工作过程描述:
* 如何a线程获得了CPU执行权, 获得了锁对象, 可以执行同步代码块;
* 在a线程执行同步代码块期间, CPU执行权被b线程抢走了 , a线程转为就绪状态;
* b线程如果想要执行同步代码块, 首先要获得锁对象, 现在锁对象在线程a手中, b线程转到等待锁对象池中,变为阻塞状态
* a线程重新获得CPU执行权, 执行完同步代码块后, 释放锁对象;
* 等待锁对象池中的线程b获得了锁对象,转为就绪状态
*/
}
package com.bjpowernode.demo04;
/**
* 创建线程类,模拟人取钱
* @author Administrator
*
*/
public class PersonThread extends Thread {
MyBankAccount myAccount ; //取钱的帐户
//构造方法
public PersonThread(MyBankAccount myAccount) {
super();
this.myAccount = myAccount;
}
@Override
public void run() {
myAccount.withdraw();
}
}
StringBuffer /Vector/Hashtable这三个类的方法都是同步方法