Maison  >  Article  >  développement back-end  >  Mise à jour par lots PHP

Mise à jour par lots PHP

不言
不言original
2018-07-06 16:23:505816parcourir

Cet article présente principalement la mise à jour par lots de PHP, qui a une certaine valeur de référence. Maintenant, je le partage avec tout le monde. Les amis dans le besoin peuvent s'y référer

La déclaration de mise à jour MySQL est très simple. d'une donnée. Un champ s'écrit généralement comme ceci :

UPDATE mytable SET myfield = 'value' WHERE other_field = 'other_value';

Si vous mettez à jour le même champ avec la même valeur, mysql est également très simple, il suffit de modifier où :

UPDATE mytable SET myfield = 'value' WHERE other_field in ('other_values');

Notez ici que other_values est une chaîne séparée par des virgules, telle que : 1,2,3

1 Solution conventionnelle
Si vous mettez à jour plusieurs éléments de données avec des valeurs différentes, de nombreuses personnes peuvent écrire comme ceci :

foreach ($display_order as $id => $ordinal) {
    $sql = "UPDATE categories SET display_order = $ordinal WHERE id = $id";
    mysql_query($sql);
}

consiste à parcourir les enregistrements de mise à jour un par un.

Un enregistrement update une fois, qui a de mauvaises performances et peut facilement provoquer un blocage.

2 Solution efficace
Les mises à jour par lots peuvent-elles donc être implémentées avec une seule instruction SQL ?

2.1 CASE WHEN
mysql ne fournit pas de méthode directe pour implémenter les mises à jour par lots, mais cela peut être réalisé avec quelques astuces.

UPDATE mytable SET
    myfield = CASE id
        WHEN 1 THEN 'value'
        WHEN 2 THEN 'value'
        WHEN 3 THEN 'value'
    END
WHERE id IN (1,2,3)

Ici, nous utilisons l'astuce case when pour réaliser des mises à jour par lots.

Par exemple :

UPDATE categories SET
    display_order = CASE id
        WHEN 1 THEN 3
        WHEN 2 THEN 4
        WHEN 3 THEN 5
    END
WHERE id IN (1,2,3)

Ce sql signifie, mettez à jour le champ display_order :

Si id=1, alors la valeur de display_order est 3,
Si id=2, la valeur de display_order est 4,
Si id=3, la valeur de display_order est 5.
signifie écrire des instructions conditionnelles ensemble.

La partie Where ici n'affecte pas l'exécution du code, mais améliorera l'efficacité de l'exécution de SQL.

Assurez-vous que l'instruction SQL n'exécute que le nombre de lignes qui doivent être modifiées. Il n'y a que 3 lignes de données à mettre à jour et la clause Where garantit que seules 3 lignes de données sont exécutées.

3.2 Mettre à jour plusieurs valeurs
Si vous souhaitez mettre à jour plusieurs valeurs, il vous suffit d'apporter de légères modifications :

UPDATE categories SET
    display_order = CASE id
        WHEN 1 THEN 3
        WHEN 2 THEN 4
        WHEN 3 THEN 5
    END,
    title = CASE id
        WHEN 1 THEN 'New Title 1'
        WHEN 2 THEN 'New Title 2'
        WHEN 3 THEN 'New Title 3'
    END
WHERE id IN (1,2,3)

À ce stade, vous avez terminé une instruction mysql à mettre à jour plusieurs enregistrements.

Mais pour l'utiliser en entreprise, il doit être combiné avec un langage côté serveur.

3.3 Encapsuler dans une fonction PHP
En PHP, nous encapsulons cette fonction dans une fonction et l'appelons directement plus tard.

Pour améliorer la convivialité, nous envisageons de gérer des scénarios plus complets.

Les données suivantes doivent être mises à jour. Nous devons mettre à jour le contenu de la table id en fonction des champs parent_id et post.

Parmi eux, la valeur de id changera et la valeur de parent_id sera la même.

$data = [
    ['id' => 1, 'parent_id' => 100, 'title' => 'A', 'sort' => 1],
    ['id' => 2, 'parent_id' => 100, 'title' => 'A', 'sort' => 3],
    ['id' => 3, 'parent_id' => 100, 'title' => 'A', 'sort' => 5],
    ['id' => 4, 'parent_id' => 100, 'title' => 'B', 'sort' => 7],
    ['id' => 5, 'parent_id' => 101, 'title' => 'A', 'sort' => 9],
];

Par exemple, nous souhaitons que les enregistrements où parent_id vaut 100 et title vaut A soient mis à jour par lots en fonction de différents identifiants :

echo batchUpdate($data, 'id', ['parent_id' => 100, 'title' => 'A']);

Parmi eux, le PHP implémenté par batchUpdate() Le code est le suivant :

/**
 * 批量更新函数
 * @param $data array 待更新的数据,二维数组格式
 * @param array $params array 值相同的条件,键值对应的一维数组
 * @param string $field string 值不同的条件,默认为id
 * @return bool|string
 */
function batchUpdate($data, $field, $params = [])
{
   if (!is_array($data) || !$field || !is_array($params)) {
      return false;
   }

    $updates = parseUpdate($data, $field);
    $where = parseParams($params);

    // 获取所有键名为$field列的值,值两边加上单引号,保存在$fields数组中
    // array_column()函数需要PHP5.5.0+,如果小于这个版本,可以自己实现,
    // 参考地址:http://php.net/manual/zh/function.array-column.php#118831
    $fields = array_column($data, $field);
    $fields = implode(',', array_map(function($value) {
        return "'".$value."'";
    }, $fields));

    $sql = sprintf("UPDATE `%s` SET %s WHERE `%s` IN (%s) %s", 'post', $updates, $field, $fields, $where);

   return $sql;
}

/**
 * 将二维数组转换成CASE WHEN THEN的批量更新条件
 * @param $data array 二维数组
 * @param $field string 列名
 * @return string sql语句
 */
function parseUpdate($data, $field)
{
    $sql = '';
    $keys = array_keys(current($data));
    foreach ($keys as $column) {

        $sql .= sprintf("`%s` = CASE `%s` \n", $column, $field);
        foreach ($data as $line) {
            $sql .= sprintf("WHEN '%s' THEN '%s' \n", $line[$field], $line[$column]);
        }
        $sql .= "END,";
    }

    return rtrim($sql, ',');
}

/**
 * 解析where条件
 * @param $params
 * @return array|string
 */
function parseParams($params)
{
   $where = [];
   foreach ($params as $key => $value) {
      $where[] = sprintf("`%s` = '%s'", $key, $value);
   }
   
   return $where ? ' AND ' . implode(' AND ', $where) : '';
}

Obtenir une telle instruction SQL de mise à jour par lots :

UPDATE `post` SET `id` = CASE `id` 
WHEN '1' THEN '1' 
WHEN '2' THEN '2' 
WHEN '3' THEN '3' 
WHEN '4' THEN '4' 
WHEN '5' THEN '5' 
END,`parent_id` = CASE `id` 
WHEN '1' THEN '100' 
WHEN '2' THEN '100' 
WHEN '3' THEN '100' 
WHEN '4' THEN '100' 
WHEN '5' THEN '101' 
END,`title` = CASE `id` 
WHEN '1' THEN 'A' 
WHEN '2' THEN 'A' 
WHEN '3' THEN 'A' 
WHEN '4' THEN 'B' 
WHEN '5' THEN 'A' 
END,`sort` = CASE `id` 
WHEN '1' THEN '1' 
WHEN '2' THEN '3' 
WHEN '3' THEN '5' 
WHEN '4' THEN '7' 
WHEN '5' THEN '9' 
END WHERE `id` IN ('1','2','3','4','5')  AND `parent_id` = '100' AND `title` = 'A'

Le SQL généré répertorie toutes les situations.

Cependant, en raison des conditions WHERE, seuls les enregistrements portant les ID 1, 2 et 3 sont mis à jour.

Si vous avez seulement besoin de mettre à jour une certaine colonne et que d'autres conditions ne sont pas limitées, alors les $data entrantes peuvent être plus simples :

$data = [
    ['id' => 1, 'sort' => 1],
    ['id' => 2, 'sort' => 3],
    ['id' => 3, 'sort' => 5],
];
echo batchUpdate($data, 'id');

Si le format des données est transmis, vous peut le modifier id Pour les enregistrements de 1 à 3, remplacez sort par 1, 3 et 5 respectivement.

Récupérez l'instruction SQL :

UPDATE `post` SET `id` = CASE `id` 
WHEN '1' THEN '1' 
WHEN '2' THEN '2' 
WHEN '3' THEN '3' 
END,`sort` = CASE `id` 
WHEN '1' THEN '1' 
WHEN '2' THEN '3' 
WHEN '3' THEN '5' 
END WHERE `id` IN ('1','2','3')

Cette situation est plus simple et plus efficace.

Ce qui précède représente l'intégralité du contenu de cet article. J'espère qu'il sera utile à l'étude de chacun. Pour plus de contenu connexe, veuillez faire attention au site Web PHP chinois !

Recommandations associées :

Comment générer rapidement des formulaires modernes avec PHP

Ce qui précède est le contenu détaillé de. pour plus d'informations, suivez d'autres articles connexes sur le site Web de PHP en chinois!

Déclaration:
Le contenu de cet article est volontairement contribué par les internautes et les droits d'auteur appartiennent à l'auteur original. Ce site n'assume aucune responsabilité légale correspondante. Si vous trouvez un contenu suspecté de plagiat ou de contrefaçon, veuillez contacter admin@php.cn