BatchUpdateService.php 3.6 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091
  1. <?php
  2. namespace App\Services\common;
  3. use App\Services\LogService;
  4. use Illuminate\Support\Facades\DB;
  5. use App\Traits\ServiceAppAop;
  6. class BatchUpdateService
  7. {
  8. use ServiceAppAop;
  9. public function batchUpdate($tableName = '', $multipleData = array(), $connection = 'mysql')
  10. {
  11. if (count($multipleData) == 1) {
  12. return false;
  13. }
  14. if ($tableName ?? false && !empty($multipleData)) {
  15. // column of fields to update
  16. $updateColumns = $multipleData[0];
  17. // table primary key
  18. $referenceColumn = $updateColumns[0];
  19. unset($multipleData[0]);
  20. if (count($multipleData) <= 200) {
  21. return $this->batchHandle($tableName, $updateColumns, $referenceColumn, $multipleData, $connection);
  22. } else {
  23. $chunk_datas = array_chunk($multipleData, 200, true);
  24. $bool = true;
  25. foreach ($chunk_datas as $chunk_data) {
  26. $is_successful = $this->batchHandle($tableName, $updateColumns, $referenceColumn, $chunk_data, $connection);
  27. $bool = $is_successful && $bool;
  28. }
  29. return $bool;
  30. }
  31. } else {
  32. return false;
  33. }
  34. }
  35. private function batchHandle($tableName, $updateColumns, $referenceColumn, $multipleData, $connection = 'mysql')
  36. {
  37. $updateSql = "UPDATE " . $tableName . " SET ";
  38. $sets = [];
  39. $bindings = [];
  40. foreach ($updateColumns as $updateColumn) {
  41. $setSql = ' `' . $updateColumn . '` = CASE';
  42. foreach ($multipleData as $multipleDatum) {
  43. $setSql .= ' WHEN ' . $referenceColumn . ' = ? THEN ? ';
  44. $bindings[] = $multipleDatum[$referenceColumn];
  45. $bindings[] = $multipleDatum[$updateColumn];
  46. }
  47. $setSql .= ' ELSE `' . $updateColumn . "` END";
  48. $sets[] = $setSql;
  49. }
  50. $updateSql .= implode(',', $sets);
  51. $whereIn = data_get($multipleData, '*.' . $referenceColumn);
  52. $bindings = array_merge($bindings, $whereIn);
  53. $whereIn = rtrim(str_repeat('?,', count($whereIn)), ',');
  54. $updateSql = rtrim($updateSql, ", ") . " WHERE `" . $referenceColumn . "` IN (" . $whereIn . ")";
  55. try {
  56. // DB::update
  57. $bool = DB::connection($connection)->update($updateSql, $bindings);
  58. app('LogService')->log(__METHOD__, __FUNCTION__, '批量更新' . count($multipleData) . $tableName . $updateSql . json_encode($bindings));
  59. return $bool;
  60. } catch (\Exception $e) {
  61. app('LogService')->log(__METHOD__, __FUNCTION__, '批量更新失败' . count($multipleData) . $tableName . $updateSql . json_encode($bindings) . $e->getMessage() . $e->getTraceAsString());
  62. return false;
  63. }
  64. }
  65. public function batchUpdateItself($tableName, $column, array $params, $connection = 'mysql')
  66. {
  67. $sql = "UPDATE ".$tableName;
  68. $sql .= " SET ".$column." = CASE ".$column;
  69. foreach ($params as $key => $value){
  70. $sql .= " WHEN ".$key." THEN ".$value;
  71. }
  72. $sql .= " END WHERE ".$column." IN (".implode(',',array_keys($params)).")";
  73. try {
  74. $bool = DB::connection($connection)->update($sql);
  75. app('LogService')->log(__METHOD__, __FUNCTION__, '批量更新本身' .$sql);
  76. return $bool;
  77. }catch (\Exception $e){
  78. app('LogService')->log(__METHOD__, __FUNCTION__, '批量更新失败 SQL:(' . $sql .") 堆栈:". $e->getMessage() . $e->getTraceAsString());
  79. return false;
  80. }
  81. }
  82. }