При обновлении дубликата ключа несколько значений MYSQL - PullRequest
0 голосов
/ 19 октября 2018

У нас есть динамический запрос, который вставляет данные из файла 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)

Этот последний, очевидно, не работает, но нам легче работать с ним, потому что у нас уже есть столбцы в одной строке иПервый запрос требует разбиения строки в массив и выполнения нескольких вложенных циклов.

//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 голосов
/ 20 октября 2018

Ваш PHP сниппет проясняет ваши намерения.

Нет, в SQL-запросе нет способа справиться с этим.

Однако вы можете просто добавить «больше магии» в PHP-скрипт для генерации требуемого запроса.

Т.е. вы уже создаете правильное INSERT -Statement из 2-х массивов.Итак, все, что вам нужно сделать, это также создать правильное ON DUPLICATE KEY -Statement.: -)

array_walk, implode и explode Ваши друзья здесь:

$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, чтобы исключить пары ключ / значение, которые вы не хотите обновлять в случае дублирования ключа.(Если есть)

...