博客
关于我
java线程(17)——Lock锁,三个线程抢票加上lock锁后变成三个线程排队买票
阅读量:322 次
发布时间:2019-03-04

本文共 1923 字,大约阅读时间需要 6 分钟。

多线程抢票问题及Lock锁的应用

1. 线程抢票问题

传统的多线程票务系统可能会出现线程安全问题,尤其是在多个线程同时操作共享资源时。例如,三个线程同时抢票可能导致票数重复售出或票数异常减少等问题。

不安全的代码示例

public class TestLock {    public static void main(String[] args) {        TestLock2 testLock2 = new TestLock2();        new Thread(testLock2).start();        new Thread(testLock2).start();        new Thread(testLock2).start();    }}class TestLock2 implements Runnable {    int ticketNums = 10;        @Override    public void run() {        while (true) {            if (ticketNums > 0) {                Thread.sleep(1000);                System.out.println(ticketNums--);            } else {                break;            }        }    }}

运行结果示例

运行时可能会出现以下问题:

  • 同一个票可能被多个线程同时抢购
  • 票数可能会异常减少
  • 2. 使用Lock锁解决问题

    通过引入ReentrantLock锁,可以有效防止线程抢票问题。以下是改进后的代码:

    import java.util.concurrent.locks.ReentrantLock;public class TestLock {    public static void main(String[] args) {        TestLock2 testLock2 = new TestLock2();        new Thread(testLock2).start();        new Thread(testLock2).start();        new Thread(testLock2).start();    }}class TestLock2 implements Runnable {    int ticketNums = 10;    private final ReentrantLock lock = new ReentrantLock();    @Override    public void run() {        while (true) {            lock.lock();            try {                if (ticketNums > 0) {                    Thread.sleep(1000);                    System.out.println(ticketNums--);                } else {                    break;                }            } finally {                lock.unlock();            }        }    }}

    改进后的运行结果

  • 每个线程将独占性地持有锁,确保票数正确递减
  • 票数将按顺序递减,避免重复售出
  • 系统运行更加稳定,线程安全得到了有效保障
  • 3. Lock锁与synchronized的对比

    主要区别

  • 锁的类型

    • Lock锁是显示锁,需要手动开启和关闭锁
    • synchronized是隐式锁,自动释放锁定状态
  • 锁的应用场景

    • Lock锁提供更细粒度的锁控制,可以灵活配置锁定区域
    • synchronized的锁定范围更大,通常用于方法或代码块层面的资源保护
  • 性能特点

    • Lock锁的锁解锁操作由用户手动控制,JVM调度线程时更高效
    • synchronized的锁解锁操作由JVM自动处理,性能相对较低
  • 选择建议

    在多线程环境下,建议根据具体需求选择合适的锁机制:

    • 如果需要细粒度的资源控制,Lock锁是一个更好的选择
    • 如果需要快速实现线程安全,synchronized提供了一个简洁的解决方案

    转载地址:http://zseq.baihongyu.com/

    你可能感兴趣的文章
    php flush()刷新不能输出缓冲的原因分析
    查看>>
    Referenced classpath provider does not exist: org.maven.ide.eclipse.launchconfig
    查看>>
    Refactoring-Imporving the Design of Exsiting Code — 代码的坏味道
    查看>>
    PHP imap 远程命令执行漏洞复现(CVE-2018-19518)
    查看>>
    php include和require
    查看>>
    ref 和out 区别
    查看>>
    php JS 导出表格特殊处理
    查看>>
    php json dom解析
    查看>>
    ReentrantReadWriteLock读写锁解析
    查看>>
    php laravel实现依赖注入原理(反射机制)
    查看>>
    php laravel请求处理管道(装饰者模式)
    查看>>
    ReentrantReadWriteLock读写锁底层实现、StampLock详解
    查看>>
    PHP mongoDB 操作
    查看>>
    ReentrantLock读写锁
    查看>>
    ReentrantLock的公平锁与非公平锁
    查看>>
    php mysql procedure获取多个结果集
    查看>>
    php mysql query 行数,PHP和MySQL:返回的行数
    查看>>
    php mysql session_php使用MySQL保存session会话
    查看>>
    PHP mysql_real_escape_string() 函数防SQL注入
    查看>>
    php mysql优化方法_MySQL优化常用方法
    查看>>