BatchUpdateService.php 3.7 KB

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