Laravel:使用同步或 updateOrCreate 更新父表和子表?

Posted

技术标签:

【中文标题】Laravel:使用同步或 updateOrCreate 更新父表和子表?【英文标题】:Laravel: Use sync or updateOrCreate for updating parent table and child table? 【发布时间】:2021-11-05 22:27:19 【问题描述】:

我有一个这样的父表:

俱乐部

id name budget
1 Arsenal 90
2 Chelsea 150
3 Man City 135
4 Man Utd 140
5 Tottenham 87

还有一个像这样的子表

玩家

id club_id name position
1 3 Grealish LM
2 3 Sterling LW
3 3 Haaland ST
4 1 Dybala ST
5 1 Casemiro DM
6 4 Fred DM
7 2 Mbappe ST
8 2 Hazard LW
9 4 Varane DM

club_id 是引用 clubs 表 id 列的外键

俱乐部模式

public function players()

    return $this->hasMany('App\Models\Player', 'club_id', 'id');

玩家模型

public function club()

    return $this->belongsTo('App\Models\Club', 'club_id', 'id');

我有一个数组输入和逻辑来创建这样的俱乐部和球员:

// the array create input will contain the club and all the players that is related to the club
$createInput = array(
  'title' => 'Liverpool',
  'budget' => '70'
  'players' => [
      [ 
         'name' => 'Handerson',
         'Position' => 'CM'
      ],
      [ 
         'name' => 'Milner',
         'Position' => 'LW'
      ]
   ]
);

$club = new Club;
$club->name = $createInput['name];
$club->budget = $createInput['name];
$club->save();
foreach($createInput['players'] as $playerInput)
  $player = new Player;
  $player->name = $playerInput['name'];
  $player->position = $playerInput['position'];
  $player->save();
]

我有一个数组输入和逻辑来更新俱乐部和球员,如下所示:

// the array update input will contain the club and all the players that are related to the club

$updateInput = array(
  'id'.  => 1,
  'budget' => '50',
  'players' => [
      [ 
         'id' => '4',
         'name' => 'Dybala',
         'Position' => 'ST'
      ],
      [ 
         'name' => 'Sane',
         'Position' => 'LW'
      ],
      [ 
         'name' => 'De Ligt',
         'Position' => 'DF'
      ]
   ]
);

$club = Club::find($updateInput['id']);
if(isset($updateInput['name'])) $club->name = $updateInput['name'];
if(isset($updateInput['budget'])) $club->budget = $updateInput['budget'];
$club->save();

$existingIds = [];
foreach($updateInput['players'] as $playerInput)
  if(isset($playerInput['id'])) $player = Player::find($playerInput['id']);
  else $player = new Player;
  if(isset($updateInput['name'])) $player->name = $playerInput['name'];
  if(isset($updateInput['position'])) $player->name = $playerInput['position'];
  $player->save();
  $existingIds[] = $player->id;


//if the player id is not in the input data then delete it 

DB::table('players')->whereNotIn('id', $existingIds)->where('club_id', $club->id)->delete();

我打算通过使用 sync 和 createorupdate 方法来简化我的代码,但我不确定哪个更适合插入/更新/删除连接到俱乐部表的相关球员数据?

【问题讨论】:

【参考方案1】:

你有一个明确的关系,那么为什么不使用它:

当您创建新记录时:

$club = new Club;
$club->name = $createInput['name];
$club->budget = $createInput['name];
$club->save();

// club_id automatic attach with players
$club->players()->createMany( $createInput['players'] );

虽然更新有关系的记录:

$club = Club::find($updateInput['id']);
if(isset($updateInput['name'])) $club->name = $updateInput['name'];
if(isset($updateInput['budget'])) $club->budget = $updateInput['budget'];
$club->save();

foreach($updateInput['players'] as $playerInput)
    $playerData = $club->players()->updateOrCreate(
        ['id' => $playerInput['id']],
        ['name' => $playerInput['name'], 'position' => $playerInput['position']]
    );

当您需要删除具有父关系的记录时:

$club = Club::find( $id_of_club );
$club->players()->delete();

【讨论】:

感谢您的回答,updateOrCreate 是否会返回一个播放器模型,因为我需要检索 id 来删除输入中不存在的播放器? @Padawan Offcourse 进入创建或更新方法调用的循环,您可以将数据分配给变量并获取 id 有没有一种方法可以专门过滤哪些玩家需要使用玩家 ID 删除,例如 $club->players()->->whereNotIn('id', $existingIds)->delete(); @Padawan 是的,但您需要创建一个逻辑,在 updateOrCreate 操作之前收集 id 并删除所有没有 updateInput id 的玩家

以上是关于Laravel:使用同步或 updateOrCreate 更新父表和子表?的主要内容,如果未能解决你的问题,请参考以下文章

updateOrCreate在Laravel Eloquent中不起作用

即使我在 laravel 中使用 updateorcreate 方法,为啥还要复制数据

在 Laravel 中使用 updateOrCreate 进行多数据插入

PHP Laravel,方法 updateOrCreate 的替换

updateOrCreate - Laravel 中的批量分配异常

我们可以将模型传递给雄辩的Laravel的updateOrCreate方法吗?