zoukankan      html  css  js  c++  java
  • php 处理并发问题

    对于商品抢购等并发场景下,可能会出现超卖的现象,这时就需要解决并发所带来的这些问题了
    PHP语言中并没有原生的提供并发的解决方案,因此就需要借助其他方式来实现并发控制。
    方案一:使用文件锁排它锁
    flock函数用于获取文件的锁,这个锁同时只能被一个线程获取到,其它没有获取到锁的线程要么阻塞,要么获取失败
    在获取到锁的时候,先查询库存,如果库存大于0,则进行下订单操作,减库存,然后释放锁
     
    方案二:使用MySQL数据库提供的悲观锁
    Innodb存储引擎支持行级锁,当某行数据被锁定时,其他进程不能对这行数据进行操作
    先查询并锁定行:select stock_num from table where id=1 for update
    if (stock_num > 0) {
        //下订单
        update table set stock_num = stock - 1 where id = 1
    }
     
    方案三:使用队列
    将用户的下单请求依次存入一个队列中,后台用一个单独的进程处理队列中的下单请求
     
    方案四:使用Redis
    redis的操作都是原子性的,可以将商品的库存存入redis中,下单之前对库存进行decr操作,如果返回的值大于等于0等可以下单,否则不能下单,这种方式效率较高
    if (redis - > get('stock_num') > 0) {
        stock_num = redis - > decr('stock_num')
        if (stock_num >= 0) {
            //下订单
        } else {
            //库存不足
        }
    } else {
        //库存不足
    }
     
    其他并发问题:
    在现实应用中,很多情况下会把数据存入缓存,当缓存失效时,去数据库取数据并重新设置缓存,如果这时并发量很大,会有很多进程同时去数据库取数据,导致很多请求
    穿透到数据库,而使数据库奔溃,这里可用文件锁来解决
    $data = $cache - > get('key');
    if (!$data) {
        $fp = fopen('lockfile');
        if (flock($fp, LOCK_EX)) {
            $data = $cache - > get('key'); //拿到锁后再次检查缓存,这时可能已经有了 
            if (!$data) {
                $data = mysql - > query();
                $cache - > set('key', $data);
            }
            flock($fp, LOCK_UN);
        }
        fclose($fp);
    }

    您的资助是我最大的动力!
    金额随意,欢迎来赏!

    如果,您认为阅读这篇博客让您有些收获,不妨点击一下右下角的推荐按钮。
    如果,您希望更容易地发现我的新博客,不妨点击一下绿色通道的关注我

    如果,想给予我更多的鼓励,求打

    因为,我的写作热情也离不开您的肯定支持,感谢您的阅读!

  • 相关阅读:
    .NET Windows服务开发流程
    SQL Server 动态行转列(参数化表名、分组列、行转列字段、字段值)
    SQL触发器批量删除数据库中的表
    如何使用Chrome Timeline 工具(译)
    修复运行 tasklist 命令时提示 ERROR: Not found
    删除坏掉的 Active Directory Domain
    为 WSUS 服务器定期运行清理向导
    为 Exchange 服务器编写自定义的反垃圾插件
    Windows 10 下安装 npm 后全局 node_modules 和 npm-cache 文件夹的设置
    在 Win10 命令行使用 Consolas + 微软雅黑
  • 原文地址:https://www.cnblogs.com/GreenForestQuan/p/6379874.html
Copyright © 2011-2022 走看看