在重复键更新多个值MYSQL

时间:2018-10-19 20:58:00

标签: mysql on-duplicate-key

我们有一个动态查询,可以从CSV文件插入数据,我们想要的是即使有重复也要更新所有行。

我们知道这可行:

INSERT INTO TABLE (ID1,ID2,ID3,PAYDATE,PRICE,CURRENCY)
VALUES ('75','2','16','2018-11-5','300','CAD'),
('75','2','17','2018-11-10','400','USD')
ON DUPLICATE KEY UPDATE 
ID1=VALUES(ID1),
ID2=VALUES(ID2),
ID3=VALUES(ID3),
PAYDATE=VALUES(PAYDATE),
PRICE=VALUES(PRICE),
CURRENCY=VALUES(CURRENCY)

但是,在创建动态查询时,这可能会变得复杂,因为在此之前我们不知道表结构。因此,我们正在寻找类似于以下内容的查询:

INSERT INTO TABLE (ID1,ID2,ID3,PAYDATE,PRICE,CURRENCY) 
VALUES ('75','2','16','2018-11-5','300','CAD'),
('75','2','17','2018-11-10','400','USD')
ON DUPLICATE KEY UPDATE
(ID1,ID2,ID3,PAYDATE,PRICE,CURRENCY)
VALUES(ID1,ID2,ID3,PAYDATE,PRICE,CURRENCY)

这最后一个显然不起作用,但是对我们来说更容易使用,因为我们已经在单个字符串中包含了列,并且第一个查询需要将字符串分解为数组并执行一些嵌套循环。 / p>

//PHP
$cols = "ID1,ID2,ID3,PAYDATE,PRICE,CURRENCY";
$vals = "('75','2','16','2018-11-5','300','CAD'),
    ('75','2','17','2018-11-10','400','USD')";

$query = "INSERT INTO TABLE ($cols) 
    VALUES $vals
    ON DUPLICATE KEY UPDATE
    ($cols) VALUES($cols)";

有类似的方法吗?

1 个答案:

答案 0 :(得分:1)

您的PHP代码段使您的意图更清晰。

否,在SQL查询中无法处理此问题。

但是,您可以简单地向PHP脚本添加“更多魔力”以生成所需的查询。

即您已经在2个数组中创建了正确的INSERT声明。因此,您所需要做的就是也创建正确的ON DUPLICATE KEY声明。 :-)

array_walkimplodeexplode是您的朋友在这里:

$cols = "ID1,ID2,ID3,PAYDATE,PRICE,CURRENCY";
$vals = "('75','2','16','2018-11-5','300','CAD'),
    ('75','2','17','2018-11-10','400','USD')";

//Create a pseudoArray, with VALUES() Keyword
$pseudoArray = explode(",", $cols);
array_walk($pseudoArray, "wrapIt");

function wrapIt(&$item, $value){
   $item = $item . "=VALUES(" . $item . ")";
}

$query = "INSERT INTO TABLE ($cols) 
    VALUES $vals
    ON DUPLICATE KEY UPDATE
    " . implode(",", $pseudoArray);


echo $query;

将产生:

INSERT INTO TABLE 
    (ID1,ID2,ID3,PAYDATE,PRICE,CURRENCY) 
VALUES 
    ('75','2','16','2018-11-5','300','CAD'), 
    ('75','2','17','2018-11-10','400','USD') 
ON DUPLICATE KEY UPDATE 
    ID1=VALUES(ID1), 
    ID2=VALUES(ID2), 
    ID3=VALUES(ID3), 
    PAYDATE=VALUES(PAYDATE), 
    PRICE=VALUES(PRICE), 
    CURRENCY=VALUES(CURRENCY)

这应该是必需的。修改wrapIt,以排除键重复的情况下不想更新的键/值对。 (如果有的话)