Java线程安全问题与同步锁

69 篇文章 2 订阅
7 篇文章 0 订阅

Java线程安全问题与同步锁

一篇博文认识Java多线程大致认识了线程也知道了线程的创建方法。今天再来探讨一下线程的安全与同步锁问题。
场景:模拟三个售票员卖30张票
一、测试继承方式创建线程

1)、线程类

package com.hk.java.thread;

/**
 * 继承创建法
 * 
 * @author 浪丶荡
 * 
 */
public class OneConductor extends Thread {

    static int tickets = 30;// 30张票

    @Override
    public void run() {// 重写run方法
        while (tickets > 0) {
            System.out.println("售票员" + Thread.currentThread().getName()
                    + "售出了第" + tickets + "张票");
            tickets--;
            //线程跑的太快了,不好观察结果,让其等待半秒
            try {
                Thread.sleep(50);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        }
    }
}

2)、测试类

@Test
    public void testThread_Thread(){
        OneConductor conductor_1 = new OneConductor();
        conductor_1.setName("张三");
        OneConductor conductor_2 = new OneConductor();
        conductor_2.setName("李四");
        OneConductor conductor_3 = new OneConductor();
        conductor_3.setName("王五");

        conductor_1.start();//张三开始卖票
        conductor_2.start();//李四开始卖票
        conductor_3.start();//王五开始卖票

    }

运行结果:

售票员 [ 张三 ]售出了第30张票
售票员 [ 张三 ]售出了第29张票
售票员 [ 张三 ]售出了第28张票
售票员 [ 张三 ]售出了第27张票
售票员 [ 张三 ]售出了第26张票
售票员 [ 张三 ]售出了第25张票
售票员 [ 张三 ]售出了第24张票
售票员 [ 张三 ]售出了第23张票
售票员 [ 张三 ]售出了第22张票
售票员 [ 张三 ]售出了第21张票
售票员 [ 张三 ]售出了第20张票
售票员 [ 王五 ]售出了第20张票
售票员 [ 王五 ]售出了第18张票
售票员 [ 王五 ]售出了第17张票
售票员 [ 王五 ]售出了第16张票
售票员 [ 王五 ]售出了第15张票
售票员 [ 王五 ]售出了第14张票
售票员 [ 王五 ]售出了第13张票
售票员 [ 王五 ]售出了第12张票
售票员 [ 王五 ]售出了第11张票
售票员 [ 王五 ]售出了第10张票
售票员 [ 王五 ]售出了第9张票
售票员 [ 王五 ]售出了第8张票
售票员 [ 王五 ]售出了第7张票
售票员 [ 王五 ]售出了第6张票
售票员 [ 王五 ]售出了第5张票
售票员 [ 王五 ]售出了第4张票
售票员 [ 张三 ]售出了第19张票
售票员 [ 王五 ]售出了第3张票
售票员 [ 张三 ]售出了第2张票
售票员 [ 王五 ]售出了第1张票
售票员 [ 李四 ]售出了第1张票

可以看出第20、第1张票被卖了两次,这显然是不行的,产生的原因就是tickets这个资源时被两个线程使用了

解决办法——加同步锁

package com.hk.java.thread;

/**
 * 继承创建法
 * 
 * @author 浪丶荡
 * 
 */
public class OneConductor extends Thread {

    static int tickets = 30;// 30张票

    @Override
    public void run() {// 重写run方法
        synchronized (this) {//代码块添加同步锁
            while (tickets > 0) {
                System.out.println("售票员 [ " + Thread.currentThread().getName()
                        + " ]售出了第" + tickets + "张票");
                tickets--;
            }
        }
    }
}

这样做了你会发现——然并卵

为什么呢?synchronized (this)这个this是获取到这个运行权的线程,而我们的线程有三个,也就是说你一扇门有三把锁,三把锁都可以打开,这还是不安全,这就要求我们加一把唯一的锁,可以防火防盗防学长的锁,这个锁就是当前类的字节码文件!

package com.hk.java.thread;

/**
 * 继承创建法
 * 
 * @author 浪丶荡
 * 
 */
public class OneConductor extends Thread {

    static int tickets = 30;// 30张票

    @Override
    public void run() {// 重写run方法
        synchronized (OneConductor.class) {//当前类的字节码作为锁
            while (tickets > 0) {
                System.out.println("售票员 [ " + Thread.currentThread().getName()
                        + " ]售出了第" + tickets + "张票");
                tickets--;
            }
        }
    }
}


看结果

售票员 [ 张三 ]售出了第30张票
售票员 [ 张三 ]售出了第29张票
售票员 [ 张三 ]售出了第28张票
售票员 [ 张三 ]售出了第27张票
售票员 [ 张三 ]售出了第26张票
售票员 [ 张三 ]售出了第25张票
售票员 [ 张三 ]售出了第24张票
售票员 [ 张三 ]售出了第23张票
售票员 [ 张三 ]售出了第22张票
售票员 [ 张三 ]售出了第21张票
售票员 [ 张三 ]售出了第20张票
售票员 [ 张三 ]售出了第19张票
售票员 [ 张三 ]售出了第18张票
售票员 [ 张三 ]售出了第17张票
售票员 [ 张三 ]售出了第16张票
售票员 [ 张三 ]售出了第15张票
售票员 [ 张三 ]售出了第14张票
售票员 [ 张三 ]售出了第13张票
售票员 [ 张三 ]售出了第12张票
售票员 [ 张三 ]售出了第11张票
售票员 [ 张三 ]售出了第10张票
售票员 [ 王五 ]售出了第9张票
售票员 [ 王五 ]售出了第8张票
售票员 [ 王五 ]售出了第7张票
售票员 [ 王五 ]售出了第6张票
售票员 [ 王五 ]售出了第5张票
售票员 [ 王五 ]售出了第4张票
售票员 [ 王五 ]售出了第3张票
售票员 [ 王五 ]售出了第2张票
售票员 [ 王五 ]售出了第1张票

经过本人多轮测试,通过继承 Thread 类创建的线程必须的使用字节码锁才能保证数据安全,而通过实现Runnable 接口创建的线程加this锁或者字节码锁都可以保证数据安全,其中缘由不是很清楚,所以在使用同步时推荐使用Runnable 接口创建的线程



如果有知道其中缘由的大牛请不吝赐教——qq群:511906138

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
同步锁Java多线程中用于保护共享资源,以确保同一时间只有一个线程可以访问该资源,从而避免数据竞争和并发问题。在Java中,可以使用synchronized关键字来实现同步锁的机制。 对于非静态的同步方法,锁可以是this对象或其他对象,要求是同一个对象。例如,使用关键字synchronized修饰的sell()方法,锁就在this对象上。 对于静态的同步方法,锁是当前类本身。因为静态方法可以在没有实例化对象的情况下使用,所以只能使用类来作为锁。可以使用synchronized修饰的静态方法m1()和m2()是示例。 除了直接在方法上使用synchronized关键字,还可以使用同步代码块来实现锁的机制。同步代码块的锁对象可以是this对象或其他对象。 当一个线程持有锁时,其他线程将无法获得该锁,它们将被阻塞,直到持有锁的线程释放锁。锁的释放可以通过以下方式实现: 1. 当前线程的同步方法或同步代码块执行结束。 2. 当前线程在同步代码块或同步方法中遇到break或return语句。 3. 当前线程在同步代码块或同步方法中出现未处理的Error或Exception,导致异常结束。 4. 当前线程在同步代码块或同步方法中执行了线程对象的wait()方法,暂停当前线程,并释放锁。 需要注意的是,线程执行同步代码块或同步方法时,调用Thread.sleep()或Thread.yield()方法暂停当前线程的执行不会释放锁。此外,使用suspend()方法将线程挂起也不会释放锁。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值