15. 线程同步-sychronized与Lock

java 多线程系列文章列表, 请查看目录: 《java 多线程学习笔记》

1. 线程安全问题

在多线程环境下, 如果使用线程不安全的类型, 可能会产生并发问题. 解决线程安全问题, 笔者总结有以下三种情况:

  • 使用线程安全的变量: Automicxxx 等
  • 使用隐形锁, 借助synchronized 关键字
  • 使用显示锁, java5 新增的Lock 接口

1.1 模拟多线程并发问题

public class Product {

    private Integer total = 10;

    public void sale(){
        if(total > 1){
            total = total -1;
            System.out.println(Thread.currentThread().getName() + "-售一件商品, 剩余商品:" + total);
        }
    }

    public static void main(String[] args) {

        // 创建多线程共享变量
        Product product = new Product();

        // 开启3个线程
        for (int i = 0; i < 3; i++) {
            new Thread(){
                @Override
                public void run() {
                    while (true) {
                        // 休眠1秒
                        ThreadUtil.sleep(1);

                        product.sale();
                    }
                }
            }.start();
        }
    }
}

1.2 测试输出

从输出结果可以发现, 商品剩余数量紊乱. 这也便是线程安全问题

Thread-1-售一件商品, 剩余商品:7
Thread-2-售一件商品, 剩余商品:7
Thread-0-售一件商品, 剩余商品:7
Thread-1-售一件商品, 剩余商品:6
Thread-2-售一件商品, 剩余商品:5
Thread-0-售一件商品, 剩余商品:4
Thread-1-售一件商品, 剩余商品:3
Thread-2-售一件商品, 剩余商品:2
Thread-0-售一件商品, 剩余商品:1

1.3 死锁问题

当两个线程相互等待对方释放同步监视器时, 便会产生死锁. 对于死锁现象, java 虚拟机并没有监测, 也没有采取措施来处理死锁情况, 因此在多线程编程时, 一定要防范出现死锁的情况.

2. synchronized-隐式锁方式

同步方法和同步代码块儿方式也称为隐式锁方式, 这是在jdk5 之前的方式.

2.1 同步方法

synchronized public void sale(){
    if(total > 1){
        total = total -1;
        System.out.println(Thread.currentThread().getName() + "-售一件商品, 剩余商品:" + total);
    }
}

2.2 同步代码块儿

public void sale(){
    synchronized (total) {
        if(total > 1){
            total = total -1;
            System.out.println(Thread.currentThread().getName() + "-售一件商品, 剩余商品:" + total);
        }
    }
}

2.3 运行输出

从输出结果看, 会发现没有线程安全问题了.

Thread-0-售一件商品, 剩余商品:9
Thread-2-售一件商品, 剩余商品:7
Thread-1-售一件商品, 剩余商品:8
Thread-0-售一件商品, 剩余商品:6
Thread-2-售一件商品, 剩余商品:5
Thread-1-售一件商品, 剩余商品:4
Thread-0-售一件商品, 剩余商品:3
Thread-2-售一件商品, 剩余商品:2
Thread-1-售一件商品, 剩余商品:1

2.4 自动释放同步监视器的锁定

  • 自动释放情况:
    • 当同步的方法/代码块儿执行结束(正常结束, 遇到break, return终止代码块儿或方法, 抛出未捕获异常Error/Exception)时, 会自动释放同步监视器
    • 当程序中执行了wait()方法时, 当前线程暂停, 并释放同步监视器
  • 不自动释放情况:
    • 当同步代码块儿/同步方法中调用了Thread.sleep(), Thread.yield()方法时, 当前线程不会释放同步监视器.
    • 当同步代码块儿/同步方法中调用了Thread.suspend()时, 不会释放. 程序中应避免直接调用suspend()方法.

3. 显示锁方式

java5 之后, 新增了Lock 类, 用于显示控制加锁和释放锁的时机. java5 定义了两个顶级接口Lock 和 ReadWriteLock, 并分别为其提供了实现类:

  • ReentrantLock: 可重入锁, 可重复加锁, 比较常用.
  • ReentrantReadWriteLocak: 读写锁

3.1 ReentrantLock 用法

  • 务必将释放锁的操作放入finally代码块儿中, 否则容易出现锁未释放问题.
  • ReentrantLock 锁具有可重入性, 也就是说一个线程可以对已加锁的ReentrantLock 锁再次加锁. ReentrantLock 对象会维持一个计数器来追中lock()的嵌套调用. 线程在每次调用lock()加锁之后, 必须调用unlock()方法释放锁.
public class Product {

    private Integer total = 10;

    // 定义锁
    private Lock lock = new ReentrantLock();

    public void sale(){
        // 添加锁
        lock.lock();

        try {
            if(total > 1){
                total = total -1;
                System.out.println(Thread.currentThread().getName() + "-售一件商品, 剩余商品:" + total);
            }
        }finally {
            // 方法结束后必须释放锁
            lock.unlock();
        }
    }

}

3.2 ReentrantReadWriteLock 用法示例

  • 对于读写锁, 笔者并没有找到一个合适的应用场景, 因此简单写一个使用示例吧.
public class Product {

    private Integer total = 10;

    private ReadWriteLock lock = new ReentrantReadWriteLock();

    public void sale(){
        // 添加锁
        lock.writeLock().lock();

        try {
            if(total > 1){
                total = total -1;
                System.out.println(Thread.currentThread().getName() + "-售一件商品, 剩余商品:" + total);
            }
        }finally {
            // 方法结束后必须释放锁
            lock.writeLock().unlock();
        }
    }

    public void queryLeft(){
        // 添加锁
        lock.readLock().lock();

        try {
            System.out.println(Thread.currentThread().getName() + "-当前剩余商品:" + total);
        }finally {
            // 方法结束后必须释放锁
            lock.readLock().unlock();
        }
    }
}
### 回答1: 在Java编程语言中,synchronized和lock都是用于实现线程同步的机制,但它们有一些区别。 synchronized是Java语言内置的关键字,用于在方法或代码块级别上实现线程同步。当一个线程进入一个synchronized方法或代码块时,它将获得该方法或代码块的锁,其他线程将被阻塞,直到该线程释放锁。synchronized锁是隐式的,即由Java虚拟机自动管理,因此使用synchronized锁的代码通常更简单。 相比之下,Lock是Java编程语言中的一个接口,它提供了一组方法来实现线程同步。与synchronized不同,Lock是显式的锁,需要程序员手动获取和释放锁。这使得Lock比synchronized更灵活,因为它允许程序员更好地控制线程的行为,如尝试获取锁的超时时间,可以在多个条件上等待等等。 总的来说,synchronized适用于简单的线程同步问题,而Lock适用于更复杂的线程同步场景。同时,由于Lock提供了更多的控制和灵活性,因此在某些情况下它可能比synchronized更高效。 ### 回答2: synchronized和lock都可以用于多线程编程中的同步机制,但是它们有一些区别。 首先,synchronized是Java语言提供的关键字,而lock是一个接口,属于Java的Lock接口族的一员。这意味着synchronized是Java语言的一部分,而lock是Java提供的一个类库。 其次,synchronized是隐式的,通过使用synchronized关键字来表示代码块的同步,而lock是显式的,通过Lock接口的实现类来实现代码块的同步。 第三,synchronized只有一种使用方式,即使用synchronized关键字对整个代码块进行同步,而lock提供了更多的灵活性和功能,例如可以实现读写分离锁、公平锁等。此外,lock还提供了tryLock()方法,可以非阻塞地尝试获取锁,而synchronized则是阻塞的。 第四,synchronized在发生异常时会自动释放锁,而lock需要手动释放锁,如果没有正确释放锁,可能导致死锁或资源泄露等问题。 第五,synchronized可以作用于代码块、方法,或者作为类锁;而lock只能作用于代码块。 总结来说,synchronized是Java语言提供的关键字,适用于普通的同步需求,而lock是一个类库接口,提供了更灵活、功能更强大的同步机制。在开发中,可以根据具体情况选择使用synchronized或lock来实现多线程的同步。 ### 回答3: synchronized和lock都是用来控制多线程访问共享资源的机制,但它们在实现上有一些区别。 首先,synchronized是Java语言提供的原生关键字,而lock是通过Lock接口及其实现类ReentrantLock来实现的。synchronized是隐式锁,它的获取和释放是由JVM自动完成的,而lock是显式锁,需要手动获取和释放。 其次,在使用上,synchronized是非公平锁,它允许不按顺序地获取锁,而lock可以选择性地使用公平锁或非公平锁。公平锁是指多个线程按照申请锁的顺序获取锁,而非公平锁是允许线程跳过等待队列直接获取锁。 另外,lock比synchronized更加灵活。它提供了一些功能,例如可中断、可轮询以及超时等待等。使用lock可以更好地控制线程的等待时间,避免线程长时间阻塞。 此外,synchronized是基于对象的锁,每个对象都有一个与之关联的监视器锁。而lock可以同时对多个对象进行加锁操作,提供了更细粒度的控制。 然而,由于synchronized是Java语言提供的原生支持,使用起来更加简单,不需要显示地获取和释放锁,而lock需要手动控制。因此,当不需要使用lock提供的额外功能时,推荐使用synchronized来简化代码的复杂度。 总之,synchronized和lock都可以实现多线程间的同步,但在使用方式、灵活性和粒度控制上有一些区别,开发者可以根据实际需求选择适合的机制。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值