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 的替换