BatchUpdateService.php 4.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100
  1. <?php
  2. namespace App\Services\common;
  3. use App\Components\ErrorPush;
  4. use Illuminate\Support\Facades\DB;
  5. use App\Traits\ServiceAppAop;
  6. class BatchUpdateService
  7. {
  8. use ServiceAppAop,ErrorPush;
  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. $debug = debug_backtrace(DEBUG_BACKTRACE_PROVIDE_OBJECT ,1)[0];
  56. unset($debug["object"]);
  57. unset($debug["args"]);
  58. try {
  59. // DB::update
  60. $bool = DB::connection($connection)->update($updateSql, $bindings);
  61. app('LogService')->log(__METHOD__, __FUNCTION__, '批量更新:' . $tableName . ' | '.json_encode($multipleData).' | 溯源:'.json_encode($debug));
  62. if (($tableName == 'authorities' || $tableName == 'menus') && env('APP_ENV')=='production'){
  63. $myfile = fopen(base_path()."\\database\\data\\{$tableName}.data", "a+");
  64. fwrite($myfile, vsprintf(str_replace('?',"'%s'",$updateSql), $bindings).";\r\n");
  65. fclose($myfile);
  66. }
  67. return $bool;
  68. } catch (\Exception $e) {
  69. $this->push(__METHOD__."->".__LINE__,"批量更新失败",$tableName . ' | '.json_encode($multipleData).' | 溯源:'.json_encode($debug) . $e->getMessage() . $e->getTraceAsString());
  70. return false;
  71. }
  72. }
  73. public function batchUpdateItself($tableName, $column, array $params, $connection = 'mysql')
  74. {
  75. $sql = "UPDATE ".$tableName;
  76. $sql .= " SET ".$column." = CASE ".$column;
  77. foreach ($params as $key => $value){
  78. $sql .= " WHEN ".$key." THEN ".$value;
  79. }
  80. $sql .= " END WHERE ".$column." IN (".implode(',',array_keys($params)).")";
  81. try {
  82. $bool = DB::connection($connection)->update($sql);
  83. app('LogService')->log(__METHOD__, __FUNCTION__, '批量更新本身' .$sql);
  84. return $bool;
  85. }catch (\Exception $e){
  86. $this->push(__METHOD__."->".__LINE__,"批量更新失败",'SQL:(' . $sql .") 堆栈:". $e->getMessage() . $e->getTraceAsString() . $e->getMessage() . $e->getTraceAsString());
  87. return false;
  88. }
  89. }
  90. }