以下所有内容均适用于InnoDB。
我觉得了解3种不同方法的速度非常重要。
有3种方法:
- INSERT:使用ON DUPLICATE KEY UPDATE插入
- 交易:您对交易中的每条记录进行更新的位置
- 案例:在这种情况下/对于UPDATE中的每个不同记录
我刚刚进行了测试,而INSERT方法对我来说比TRANSACTION方法快6.7倍。我尝试了3,000行和30,000行的组合。
TRANSACTION方法仍然必须运行每个单独的查询,这需要花费时间,尽管它会在执行时将结果批处理到内存中。在复制和查询日志中,TRANSACTION方法也非常昂贵。
更糟糕的是,CASE方法的速度比INSERT方法慢31.1倍,具有30,000条记录(比TRANSACTION慢6.1倍)。而MyISAM的速度要慢75倍。INSERT和CASE方法收支平衡,达到约1,000条记录。即使有100条记录,CASE方法也快得多。
因此,总的来说,我觉得INSERT方法既是最好的也是最容易使用的。这些查询较小,更易于阅读,仅占用1个操作查询。这适用于InnoDB和MyISAM。
奖励的东西:
INSERT非默认字段问题的解决方案是暂时关闭相关的SQL模式:SET SESSION sql_mode=REPLACE(REPLACE(@@SESSION.sql_mode,"STRICT_TRANS_TABLES",""),"STRICT_ALL_TABLES","")
。sql_mode
如果计划还原,请确保保存第一个。
至于其他评论,我已经看到说使用INSERT方法提高了auto_increment,在InnoDB中似乎确实如此,但MyISAM却不是。
运行测试的代码如下。它还输出.SQL文件以消除php解释器的开销
<?
//Variables
$NumRows=30000;
//These 2 functions need to be filled in
function InitSQL()
{
}
function RunSQLQuery($Q)
{
}
//Run the 3 tests
InitSQL();
for($i=0;$i<3;$i++)
RunTest($i, $NumRows);
function RunTest($TestNum, $NumRows)
{
$TheQueries=Array();
$DoQuery=function($Query) use (&$TheQueries)
{
RunSQLQuery($Query);
$TheQueries[]=$Query;
};
$TableName='Test';
$DoQuery('DROP TABLE IF EXISTS '.$TableName);
$DoQuery('CREATE TABLE '.$TableName.' (i1 int NOT NULL AUTO_INCREMENT, i2 int NOT NULL, primary key (i1)) ENGINE=InnoDB');
$DoQuery('INSERT INTO '.$TableName.' (i2) VALUES ('.implode('), (', range(2, $NumRows+1)).')');
if($TestNum==0)
{
$TestName='Transaction';
$Start=microtime(true);
$DoQuery('START TRANSACTION');
for($i=1;$i<=$NumRows;$i++)
$DoQuery('UPDATE '.$TableName.' SET i2='.(($i+5)*1000).' WHERE i1='.$i);
$DoQuery('COMMIT');
}
if($TestNum==1)
{
$TestName='Insert';
$Query=Array();
for($i=1;$i<=$NumRows;$i++)
$Query[]=sprintf("(%d,%d)", $i, (($i+5)*1000));
$Start=microtime(true);
$DoQuery('INSERT INTO '.$TableName.' VALUES '.implode(', ', $Query).' ON DUPLICATE KEY UPDATE i2=VALUES(i2)');
}
if($TestNum==2)
{
$TestName='Case';
$Query=Array();
for($i=1;$i<=$NumRows;$i++)
$Query[]=sprintf('WHEN %d THEN %d', $i, (($i+5)*1000));
$Start=microtime(true);
$DoQuery("UPDATE $TableName SET i2=CASE i1\n".implode("\n", $Query)."\nEND\nWHERE i1 IN (".implode(',', range(1, $NumRows)).')');
}
print "$TestName: ".(microtime(true)-$Start)."<br>\n";
file_put_contents("./$TestName.sql", implode(";\n", $TheQueries).';');
}