
在 laravel 中,formrequest 的验证逻辑在控制器执行前就已完成,因此不能在控制器中修改请求数据后再调用 `validated()` 获取新字段;必须在验证流程早期(如 `prepareforvalidation`)注入字段,才能使其被规则识别并返回。
Laravel 的 FormRequest 类在进入控制器方法前,会自动触发完整的验证生命周期:包括预处理(prepareForValidation)、规则校验(rules)、后置处理(withValidator)等阶段。这意味着——任何在控制器中对 $request 实例的修改(例如 $this['locale_id'] = 1),都不会影响已执行完毕的验证结果,validated() 返回的始终是验证阶段最终确认的数据快照。
要让 locale_id 正确出现在 validated() 结果中,必须在验证开始前将其注入请求载荷。Laravel 官方推荐的方式是重写 prepareForValidation() 方法:
// app/Http/Requests/InvoiceCreateRequest.php
protected function prepareForValidation()
{
if ($this->has('locale_code')) {
$localeCode = $this->input('locale_code');
if (!empty($localeCode)) {
$locale = \App\Models\Locale::where('code', $localeCode)->firstOrFail();
$this->merge(['locale_id' => $locale->id]);
} else {
$this->merge(['locale_id' => null]);
}
}
}同时,确保你的验证规则明确包含该字段,并根据业务逻辑设置合适约束:
public function rules()
{
return [
'locale_id' => 'required|exists:locales,id',
'billing_first_name' => 'required|string|max:255',
'billing_last_name' => 'required|string|max:255',
];
}⚠️ 注意事项:
- prepareForValidation() 在 rules() 之前执行,因此注入的 locale_id 能被后续规则(如 exists)正常校验;
- 不要使用 $this['key'] = value 或 array_merge() 直接操作请求属性——这些不会同步到验证上下文;
- 若需兼容 locale_code 可选(即允许不传),应将 locale_id 规则设为 'nullable|exists:locales,id' 并在 prepareForValidation 中显式补 null;
- 避免在 convertLocaleCodeToLocaleId() 这类自定义方法中直接修改 $this 数组,它无法参与验证生命周期。
✅ 最终效果:调用 $request->validated() 将稳定返回包含 'locale_id' => 1 的数组,且该值已通过数据库存在性校验,可安全用于后续模型创建或业务逻辑。










