0

我这里有一个与 mysql 和 php 相关的奇怪错误。我想知道这是否也可能是我们服务器的性能问题。

我有一个用于管理回扣促销代码的课程。代码很棒,工作正常,完全符合它的预期。saveChanges() 操作根据对象的状态发送 INSERT 或 UPDATE ,并且在当前上下文中只会插入因为我正在尝试生成优惠券代码。

该类的 saveChanges 是这样的:(我知道,我不应该使用旧的 mysql,但由于软件的架构限制,我别无选择,所以请不要抱怨那部分)

public function saveChanges($asNew = false){

    //Get the connection
    global $conn_panier;

    //Check if the rebate still exists
    if($this->isNew() || $asNew){

        //Check unicity if new
        if(reset(mysql_fetch_assoc(mysql_query('SELECT COUNT(*) FROM panier_rabais_codes WHERE code_coupon = "'.mysql_real_escape_string($this->getCouponCode(), $conn_panier).'"', $conn_panier))) > 0){
            throw new Activis_Catalog_Models_Exceptions_ValidationException('Coupon code "'.$this->getCouponCode().'" already exists in the system', $this, __METHOD__, $this->getCouponCode());
        }

        //Update the existing rebate
        mysql_query($q = 'INSERT INTO panier_rabais_codes 
                        (
                            `no_rabais`,
                            `code_coupon`,
                            `utilisation`,
                            `date_verrou`
                        )VALUES(
                            '.$this->getRebate()->getId().',
                            "'.mysql_real_escape_string(stripslashes($this->getCouponCode()), $conn_panier).'",
                            '.$this->getCodeUsage().',
                            "'.($this->getInvalidityDate() === NULL ? '0000-00-00 00:00:00' : date('Y-m-d G:i:s', strtotime($this->getInvalidityDate()))).'"
                        )', $conn_panier);
        return (mysql_affected_rows($conn_panier) >= 1);

    }else{

        //Update the existing rebate
        mysql_query('UPDATE panier_rabais_codes
                        SET
                            `utilisation` = '.$this->getCodeUsage().',
                            `date_verrou` = "'.($this->getInvalidityDate() === NULL ? '0000-00-00 00:00:00' : date('Y-m-d G:i:s', strtotime($this->getInvalidityDate()))).'"
                        WHERE 
                            no_rabais = '.$this->getRebate()->getId().' AND code_coupon = "'.mysql_real_escape_string($this->getCouponCode(), $conn_panier).'"', $conn_panier);
        return (mysql_affected_rows($conn_panier) >= 0);

    }

}

如您所见,代码本身非常简单干净,如果插入成功则返回 true,否则返回 false。

代码的另一部分使用随机算法生成代码,如下所示:

while($codes_to_generate > 0){

    //Sleep to delay mysql choking on the input
    usleep(100);

    //Generate a random code
    $code = strtoupper('RC'.$rebate->getId().rand(254852, 975124));
    $code .= strtoupper(substr(md5($code), 0, 1));
    $rebateCode = new Activis_Catalog_Models_RebateCode($rebate);
    $rebateCode->setCouponCode($code);
    $rebateCode->setCodeUsage($_REQUEST['utilisation_generer']);
    try{
        if($rebateCode->saveChanges()){
            $codes_to_generate--;
            $generated_codes[] = $code;
        }
    }catch(Exception $ex){
    }

}

正如您在此处看到的,有两点需要注意。只有当我从 saveChanges 获得返回 true 时,才会填充要生成的代码数量和生成的代码数组,因此 mysql 必须报告已插入信息以使这部分发生。

另一个花絮是 while 的第一行:

//Sleep to delay mysql choking on the input
usleep(100);

什么?那么这篇文章就是关于这个的。我的代码可以完美运行,只需生成少量代码。但是,如果我要求 mysql 一次保存多个代码,我必须使用 usleep 来限制自己,否则 mysql 会删除其中一些行。它将报告有受影响的行但不保存它们。

在 100 行以下,我不需要节流,然后我需要根据要插入的行数来休眠。这一定很简单,但我不知道是什么。这是我尝试插入的行的总和以及我必须实现的最小睡眠节流阀:

  • < 100 行:无
  • < 300 行:2 毫秒
  • < 1000 行:5 毫秒
  • < 2000 行:10 毫秒
  • < 5000 行:20 毫秒
  • < 10000 行:100 毫秒

感谢您的时间

4

1 回答 1

2

您确定您的代码都已插入且未更新,因为更新不存在的行没有任何作用。

于 2012-05-28T12:53:36.010 回答