Java多线程中的对象互斥锁

1、为什么会有锁?

在看线程同步的问题之前,我们先看一个生活中的小例子:
我拿着银行卡去ATM取钱,假如我的卡里有3000块,我要取走2000,这个时候,ATM会去银行的数据库里查询我的账户是否有2000以上的余额,如果有,就会让我取走,不幸的是,这个时候,我女朋友也来银行取钱,只不过她在前台取钱,她直接取走了3000,这个时候我的卡里已经没钱了,而我也肯定也不能取出2000了。

我们可以这样想,上面例子中的我和女朋友是两个线程,我们在访问同一块资源,也就是一个账户,这个情况下,是不允许在一个线程访问的时候,另一个线程来打断的,就像我在取钱的时候,我女朋友是不能取的,不能打断我。

我们先来看一段代码:

public class TestThread01 implements Runnable{

Timer timer = new Timer();
public static void main(String[] args) {
    TestThread01 t = new TestThread01();
    Thread t1 = new Thread(t);
    Thread t2 = new Thread(t);
    t1.setName("t1");
    t2.setName("t2");
    t1.start();
    t2.start();
}
public void run(){
    timer.add(Thread.currentThread().getName());
}
}
class Timer{
private static int num = 0;
public static void add(String name){
    num++;
    try {
        Thread.sleep(1);
    } catch (InterruptedException e) {

    }
    System.out.println(name +":你是第"+num+"使用timer的线程");
}
}

上面的代码我们就能看出两个线程去访问同一个资源,也就是timer的add方法,看一下运行结果:

t1:你是第2使用timer的线程
t2:你是第2使用timer的线程

结果为什么是这样呢?

我们分析一下:
t1启动,访问add方法,num++变为1,这时,t1睡眠,t2启动,t2访问add,num是1,++之后变为2,t2睡眠,t1再继续访问,打印“t1:你是第2使用timer的线程”,然后t1结束,t2继续打印“t2:你是第2使用timer的线程”。

我们这就见到了线程的打断,有的人可能会说是因为sleep了一毫秒,所以才会这样,其实不然,如果没有sleep,也有可能会被打断,也有可能会正确执行。

2、对象互斥锁的用法?

解决方法01:给可能竞争的资源对象加上互斥锁,synchronized (this),锁定当前对象。

class Timer{
private static int num = 0;
public void add(String name){
    synchronized (this) {
    num++;
    try {
        Thread.sleep(1);
    } catch (InterruptedException e) {

    }
    System.out.println(name +":你是第"+num+"使用timer的线程");
}
}
}

这时的执行结果:

 t1:你是第1使用timer的线程
 t2:你是第2使用timer的线程

解决方法02:给竞争的方法加上锁

class Timer{
private static int num = 0;
public synchronized void add(String name){

    num++;
    try {
        Thread.sleep(1);
    } catch (InterruptedException e) {

    }
    System.out.println(name +":你是第"+num+"使用timer的线程");

}
}

结果也是对的。

    原文作者:java锁
    原文地址: https://blog.csdn.net/jin970505/article/details/78995005
    本文转自网络文章,转载此文章仅为分享知识,如有侵权,请联系博主进行删除。
点赞