资源竞争和锁

来源:互联网 发布:支持mac的对战平台 编辑:程序博客网 时间:2024/05/18 04:59

来看一个例子:

public class GetMoney implements Runnable {private int money= 100;//你的账户余额private int smoney = 0;//一共取钱的总额public void run() {while(money>0){try {Thread.currentThread().sleep(1000);} catch (InterruptedException e) {// TODO Auto-generated catch blocke.printStackTrace();}smoney += 10;money -= 10;System.out.println(Thread.currentThread().getName()+"取了一次10元,一共取了" + smoney+"元");}}public static void main(String[] args) {GetMoney gm = new GetMoney ();new Thread(gm).start();new Thread(gm).start();new Thread(gm).start();}}
这是一次运行结果:

Thread-0取了一次10元,一共取了20元
Thread-1取了一次10元,一共取了20元
Thread-2取了一次10元,一共取了20元
Thread-1取了一次10元,一共取了30元
Thread-2取了一次10元,一共取了40元
Thread-0取了一次10元,一共取了50元
Thread-1取了一次10元,一共取了60元
Thread-2取了一次10元,一共取了70元
Thread-0取了一次10元,一共取了80元
Thread-1取了一次10元,一共取了90元
Thread-2取了一次10元,一共取了100元
Thread-0取了一次10元,一共取了110元
Thread-1取了一次10元,一共取了120元

_

可以发现,金额是100,可是取出来的钱却可以多于100。因为CPU在一个时间点只能执行一个线程,CPU速度快,多个线程像是同时运行的,至于谁先执行就看哪个线程先抢到了CPU的时间片,谁先抢到谁先执行,且只执行一次,执行结束再去抢时间片。“钱”这个资源在几个窗口取的时候可以同时被访问,这就造成了可以多取的现象。我们肯定要做到资源在同一时间只能有一个线程访问。我们可以采用同步方法,用synchronized修饰方法,整个方法的代码都是同步的,只能一个线程运行。同步方法使用this作为锁。
还可以采用同步代码块,同步代码块中的内容同一时间内只能执行一个线程。同步代码块形式如下:
synchronized(锁对象—临界资源){
中间是要同步的代码
}