我这里有一个与 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 毫秒
感谢您的时间