Lock锁

Lock锁

经典卖票例子

传统Synchronized来实现。

package com.cc.demo1;

/**
 * 真正的多线程开发,公司中的开发,降低耦合性
 * 线程就是一个单独的资源类,没有任何附属的操作!
 * 1、 属性、方法
 */
public class SaleTicketDemo1 {
    public static void main(String[] args) {

        // 并发:多线程操作同一个资源类, 把资源类丢入线程
        Ticket ticket = new Ticket();

        new Thread(()->{
            for (int i = 1; i < 40 ; i++) {
                ticket.sale();
            }
        },"A").start();

        new Thread(()->{
            for (int i = 1; i < 40 ; i++) {
                ticket.sale();
            }
        },"B").start();

        new Thread(()->{
            for (int i = 1; i < 40 ; i++) {
                ticket.sale();
            }
        },"C").start();
    }
}

// 资源类 OOP
class Ticket {
    // 属性、方法
    private int number = 30;
    // 卖票的方式
    // synchronized 本质: 队列,锁
    public synchronized void sale(){
        if (number>0){
            System.out.println(Thread.currentThread().getName()+"卖出了"+(number--)+"票,剩余:"+number);
        }
    }
}

结果:

A卖出了30,剩余:29
A卖出了29,剩余:28
A卖出了28,剩余:27
A卖出了27,剩余:26
A卖出了26,剩余:25
A卖出了25,剩余:24
A卖出了24,剩余:23
A卖出了23,剩余:22
A卖出了22,剩余:21
A卖出了21,剩余:20
A卖出了20,剩余:19
A卖出了19,剩余:18
A卖出了18,剩余:17
A卖出了17,剩余:16
A卖出了16,剩余:15
A卖出了15,剩余:14
A卖出了14,剩余:13
A卖出了13,剩余:12
A卖出了12,剩余:11
A卖出了11,剩余:10
A卖出了10,剩余:9
A卖出了9,剩余:8
A卖出了8,剩余:7
A卖出了7,剩余:6
A卖出了6,剩余:5
A卖出了5,剩余:4
A卖出了4,剩余:3
A卖出了3,剩余:2
A卖出了2,剩余:1
A卖出了1,剩余:0

Lock接口来实现

可以看JDK1.8的帮助文档。

package com.cc.demo1;

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

public class SaleTicketDemo02 {
    public static void main(String[] args) {
        // 并发:多线程操作同一个资源类, 把资源类丢入线程
        Ticket2 ticket = new Ticket2();

        new Thread(()->{for (int i = 1; i < 40 ; i++)
            ticket.sale();},"A").start();
        new Thread(()->{for (int i = 1; i < 40 ; i++)
            ticket.sale();},"B").start();
        new Thread(()->{for (int i = 1; i < 40 ; i++)
            ticket.sale();},"C").start();
    }
}

// Lock三部曲
// 1、 new ReentrantLock();
// 2、 lock.lock(); // 加锁
// 3、 finally=> lock.unlock(); // 解锁
class Ticket2 {
    // 属性、方法
    private int number = 30;
    Lock lock = new ReentrantLock();

    public void sale(){
        lock.lock(); // 加锁
        try {
            // 业务代码
            if (number>0){
                System.out.println(Thread.currentThread().getName()+"卖出了"+ (number--)+"票,剩余:"+number);
            }
        } catch (Exception e) {
            e.printStackTrace();
        } finally {
            lock.unlock(); // 解锁
        }
    }
}

Synchronized和Lock区别

1、Synchronized 内置的Java关键字, Lock 是一个Java类

2、Synchronized 无法判断获取锁的状态,Lock 可以判断是否获取到了锁

3、Synchronized 会自动释放锁,lock 必须要手动释放锁!如果不释放锁,死锁

4、Synchronized 线程 1(获得锁,阻塞)、线程2(等待,傻傻的等);Lock锁就不一定会等待下 去;

5、Synchronized 可重入锁,不可以中断的,非公平;Lock ,可重入锁,可以 判断锁,非公平(可以 自己设置);

6、Synchronized 适合锁少量的代码同步问题,Lock 适合锁大量的同步代码!

ThinkPHP的机制主要是为了处理并发环境下数据的一致性和防止资源的冲突。在ThinkPHP中,可以使用Lock类来实现机制,主要支持两种类型的:文件和Redis。 文件通过创建临时文件来定资源,适用于不支持操作的环境。它的使用非常简单,只需要实例化Lock类并调用lock方法即可。文件机制会创建一个文件,只有当该文件不存在时,才表示获取成功。 Redis则是利用Redis数据库的特性来实现机制,适用于支持Redis的环境。ThinkPHP的Redis支持设置的过期时间,这样即使在程序异常退出的情况下,也能在指定时间后自动释放,避免造成死。使用Redis时,需要先配置好Redis客户端,然后实例化Lock类并调用lock方法时指定类型为'redis'。 两种的具体使用示例如下: ```php // 文件使用示例 $lock = new \think\lock\FileLock('lockfile.lock'); if ($lock->lock()) { // 成功获取后执行的操作 $lock->unlock(); } // Redis使用示例 $config = [ 'type' => 'redis', 'host' => 'localhost', // Redis服务器地址 'port' => 6379, // Redis服务器端口 'password' => '', // Redis服务器密码,如果设置了的话 'expire' => 10, // 的过期时间 ]; $redis = new \think\cache\driver\Redis($config); $lock = new \think\lock\RedisLock($redis, 'redis_lock_key'); if ($lock->lock()) { // 成功获取后执行的操作 $lock->unlock(); } ``` 注意,在使用机制时,需要确保在操作结束后调用unlock方法释放,否则会导致其他进程或线程无法获取到
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值