PHP并发操作下的加锁

代码如下:

<?php
/**
 * CacheLock 进程锁,主要用来进行cache失效时的单进程cache获取,防止过多的SQL请求穿透到数据库
 * 用于解决PHP在并发时候的锁控制,通过文件/eaccelerator进行进程间锁定
 * 如果没有使用eaccelerator则进行进行文件锁处理,会做对应目录下产生对应粒度的锁
 * 使用了eaccelerator则在内存中处理,性能相对较高
 * 不同的锁之间并行执行,类似mysql innodb的行级锁
 * 本类在sunli的phplock的基础上做了少许修改  http://code.google.com/p/phplock 
 * @author yangxinqi
 *
 */
class b2c_test_cachelock extends b2c_api_rpc_request{
    //文件锁存放路径
    private $path = null;
    //文件句柄
    private $fp = null;
    //锁粒度,设置越大粒度越小
    private $hashNum = 100;
    //cache key 
    private $name;
    //是否存在eaccelerator标志
    private  $eAccelerator = false;
    public $num = 10;
    
    /**
     * 构造函数
     * 传入锁的存放路径,及cache key的名称,这样可以进行并发
     * @param string $path 锁的存放目录,以"/"结尾
     * @param string $name cache key
     */
    public function __construct($name,$path='lock\\')
    {
        //echo "构造方法被执行......";
        //echo $name;
        //echo $path;
        $name = "key_name";
        $path = "D:\\WWW\\ecstore\\data\\lock\\";
        //判断是否存在eAccelerator,这里启用了eAccelerator之后可以进行内存锁提高效率
        $this->eAccelerator = function_exists("eaccelerator_lock");
        if(!$this->eAccelerator)
        {
            $this->path = $path.($this->_mycrc32($name) % $this->hashNum).'.txt';
            //echo "fine name:".$this->path."<br>";
        }
        $this->name = $name;
    }
    
    /**
     * crc32
     * crc32封装
     * @param int $string
     * @return int
     */
    private function _mycrc32($string)
    {
        $crc = abs (crc32($string));
        if ($crc & 0x80000000) {
            $crc ^= 0xffffffff;
            $crc += 1;
        }
        return $crc;
    }
    /**
     * 加锁
     * Enter description here ...
     */
    public function lock()
    {
        //如果无法开启ea内存锁,则开启文件锁
        if(!$this->eAccelerator)
        {
            //配置目录权限可写
            $this->fp = fopen($this->path, 'w+');
            if($this->fp === false)
            {
                return false;
            }
            return flock($this->fp, LOCK_EX);
        }else{
            return eaccelerator_lock($this->name);
        }
    }
    
    /**
     * 解锁
     * Enter description here ...
     */
    public function unlock()
    {
        if(!$this->eAccelerator)
        {
            if($this->fp !== false)
            {
                flock($this->fp, LOCK_UN);
                clearstatcache();
            }
            //进行关闭
            fclose($this->fp);
        }else{
            return eaccelerator_unlock($this->name);
        }
    }
}

?>
 

 

 

使用如下:

    //测试线程并发----测试通过解决线程并发问题
    public function fileLock(){
        $lock = kernel::single("b2c_test_cachelock");//实例化类也可以这样:$lock = new CacheLock('key_name',"D:\\WWW\\ecstore\\data\\lock\\");
        $lock->lock();//文件加锁
        //logic  如下:
        //链接数据库
        //数据库sdb_b2c_cart_objects表测试记录:obj_ident=1    member_ident=10    member=10    obj_type=goods    params=21    quantity=100    
        $conn = Mysql::mysqlConnection();//获取数据连接,这里连接你们自己的数据库
        $sql = "select quantity from sdb_b2c_cart_objects where obj_ident = 1 ";
        $result = mysql_query($sql,$conn);
        
        $quantity = 0;
        while($row = mysql_fetch_array($result,MYSQL_ASSOC)){
            $quantity= $row['quantity'];
        }
        sleep(1);//休眠一秒钟
        //logic here
        if($quantity > 0){
            //echo $lock->num;
            
            $quantity -- ;
            ToolKit::_myLogln("数量:",$quantity,LOG_DIR."fileLock");//打印日志,可以删除
            echo $quantity;
            $update_sql = 'update sdb_b2c_cart_objects set quantity = "'.$quantity.'" where obj_ident = 1';
            mysql_query($update_sql);
        }
        $lock->unlock();//文件解锁
        //使用过程中需要注意下文件锁所在路径需要有写权限.
    }

 

 

转自这篇博客:http://www.nowamagic.net/php/php_LockInConcurrent.php

 

转载于:https://my.oschina.net/qimhkaiyuan/blog/789841

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值