6.JUC线程高级-Lock 同步锁

简介: 1. 解决多线程安全问题的方式:同步代码块synchronized(this) {}同步方法 public synchronized void method() { }public static synchronized void method() { }JDK1.5之前才使用上述两种方式借助于:synchronized 隐式锁。

1. 解决多线程安全问题的方式:

  1. 同步代码块
synchronized(this) {
}
  1. 同步方法
 public synchronized void method() {
 }

public static synchronized void method() {
 }

JDK1.5之前才使用上述两种方式借助于:synchronized 隐式锁。之后出现一个新的显示同步锁

  1. 同步锁 Lock 显示锁
    显示锁:必须通过 lock() 方法上锁,通过 unlock() 方法进行释放锁
    此种方式是一种更加灵活更加高级处理线程安全问题的方式,但它也存在一定的不足,需要手动(finally)释放锁。

下面使用卖票实例来模拟 Lock锁的使用:


package com.pyy.juc;


public class TestLock {
    public static void main(String[] args) {
        Ticket ticket = new Ticket();

        new Thread(ticket, "1号窗口").start();
        new Thread(ticket, "2号窗口").start();
        new Thread(ticket, "3号窗口").start();
    }

}

class Ticket implements Runnable {

    private int tick = 100;

    @Override
    public void run() {
        while(true) {
            if(tick > 0) {
                try {
                    Thread.sleep(200);
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
                System.out.println(Thread.currentThread().getName() + " 完成售票,余票为:" + --tick);
            }
        }
    }
}

结果: 出现多个窗口卖同一张票的情况

...
2号窗口 完成售票,余票为:24
1号窗口 完成售票,余票为:23
2号窗口 完成售票,余票为:22
3号窗口 完成售票,余票为:22
1号窗口 完成售票,余票为:21
3号窗口 完成售票,余票为:20
2号窗口 完成售票,余票为:19
...

下面使用同步锁Lock- ReenTrantLock实现:


package com.pyy.juc;


import java.util.concurrent.locks.Lock;
import java.util.concurrent.locks.ReentrantLock;

public class TestLock {
    public static void main(String[] args) {
        Ticket ticket = new Ticket();

        new Thread(ticket, "1号窗口").start();
        new Thread(ticket, "2号窗口").start();
        new Thread(ticket, "3号窗口").start();
    }

}

class Ticket implements Runnable {

    private int tick = 100;

    private Lock lock = new ReentrantLock();

    @Override
    public void run() {
        while(true) {

            lock.lock();// 加锁
            
            try {
                if(tick > 0) {
                    try {
                        Thread.sleep(200);
                    } catch (InterruptedException e) {
                        e.printStackTrace();
                    }
                    System.out.println(Thread.currentThread().getName() + " 完成售票,余票为:" + --tick);
                }
            } finally {
                lock.unlock();// 释放锁
            }
        }
    }
}

结果不会出现线程安全问题。

目录
相关文章
|
1月前
|
安全 编译器 C#
C#学习相关系列之多线程---lock线程锁的用法
C#学习相关系列之多线程---lock线程锁的用法
|
3月前
|
存储 Java 数据安全/隐私保护
【JUC】ThreadLocal 如何实现数据的线程隔离?
【1月更文挑战第15天】【JUC】ThreadLocal 如何实现数据的线程隔离?ThreadLocal 导致内存泄漏问题?
|
3月前
|
安全 算法 Java
剑指JUC原理-19.线程安全集合(上)
剑指JUC原理-19.线程安全集合
25 0
|
3月前
|
Java Linux 调度
剑指JUC原理-7.线程状态与ReentrantLock(中)
剑指JUC原理-7.线程状态与ReentrantLock
34 0
|
3月前
|
存储 安全 Java
剑指JUC原理-4.共享资源和线程安全性(上)
剑指JUC原理-4.共享资源和线程安全性
44 1
|
3月前
|
监控 Java 应用服务中间件
剑指JUC原理-3.线程常用方法及状态(下)
剑指JUC原理-3.线程常用方法及状态
57 0
|
3月前
|
Java Linux API
剑指JUC原理-2.线程
剑指JUC原理-2.线程
41 0
|
3月前
|
并行计算 Java 应用服务中间件
剑指JUC原理-1.进程与线程
剑指JUC原理-1.进程与线程
23 0
|
1月前
|
安全 Java C++
JUC(java.util.concurrent)的常见类(多线程编程常用类)
JUC(java.util.concurrent)的常见类(多线程编程常用类)
|
2月前
|
安全 Java
多线程(进阶三:JUC)
多线程(进阶三:JUC)
45 0