Laravel中使用FormRequest进行表单验证方法及问题汇总

Laravel中使用FormRequest进行表单验证方法及问题汇总,第1张

概述Laravel 5.0 带来了FormRequests, 这是一种特殊的类型, 用于在提交表单时进行数据的检查和验证. 每个FormRequest类至少包含一个rules()方法, 这个方法返回一组验证规则. 除此之外还必须包含一个authorize()方法, 该方法返回一个布尔值, 代表是否允许用户执行本次请求.

在`Laravel`中,每一个请求都会被封装为一个`Request`对象,`Form Request`对象就是包含了额外验证逻辑(以及访问权限控制)的自定义`Request`类。 本文分析了FormRequest异常的处理流程并提出了自定义处理FormRequest验证失败的思路。

所有示例基于Laravel 5.1.39 (LTS)

今天天气不错,我们来说说表单验证。

Controller中做表单验证

有的同学把表单验证逻辑写在Controller中,例如这个对用户提交评论内容的验证:

// ...

use Validator;

class CommentController
{

public function postStoreComment(Request $request)
{
$validator = Validator::make($request->all(),[
'comment' => 'required',// 只是实例,就写个简单的规则,你的网站要是这么写欢迎在评论里贴网址
]);

if ($val<a href="https://m.jb51.cc/tag/ID/" target="_blank" >ID</a>ator->fails()) {  return redirect()    ->back()    ->withErrors($val<a href="https://m.jb51.cc/tag/ID/" target="_blank" >ID</a>ator)    ->with<a href="https://www.jb51.cc/tag/input/" target="_blank" >input()</a>;}

}

这样写的话,表单验证和业务逻辑挤在一起,我们的Controller中就会有太多的代码,而且重复的验证规则基本也是复制粘贴。

我们可以利用Form Request来封装表单验证代码,从而精简Controller中的代码逻辑,使其专注于业务。而独立出去的表单验证逻辑甚至可以复用到其它请求中,例如修改评论。

什么是Form Request

在Laravel中,每一个请求都会被封装为一个Request对象,Form Request对象就是包含了额外验证逻辑(以及访问权限控制)的自定义Request类。

如何使用Form Request做表单验证

Laravel提供了生成Form Request的Artisan命令:

$ PHP artisan make:request StoreCommentRequest

于是就生成了app/http/Requests/StoreCommentRequest.PHP,让我们来分析一下内容:

namespace App\http\Requests;

use App\http\Requests\Request; // 可以看到,这个基类是在我们的项目中的,这意味着我们可以修改它

class StoreCommentRequest extends Request
{
/**

Determine if the user is authorized to make this request.@return bool
*/
public function authorize() // 这个方法可以用来控制访问权限,例如禁止未付费用户评论…
{
return false; // 注意!这里默认是false,记得改成true
}

/**

Get the valIDation rules that apply to the request.@return array
*/
public function rules() // 这个方法返回验证规则数组,也就是ValIDator的验证规则
{
return [
//
];
}
}

那么很容易,我们除了让authorize方法返回true之外,还得让rules方法返回我们的验证规则:

// ...

public function rules()
{
return [

];

}

// ...

接着修改我们的Controller:

// ...

// 之前:public function poststoreComment(Request $request)
public function poststoreComment(\App\http\Requests\StoreCommentRequest $request)
{
// ...
}

// ...

这样Laravel便会自动调用StoreCommentRequest进行表单验证了。

异常处理

如果表单验证失败,Laravel会重定向到之前的页面,并且将错误写到Session中,如果是AJAX请求,则会返回一段http状态为422的JsON数据,类似这样:

{comment: ["The comment fIEld is required."]}

这里就不细说提示信息怎么修改了,如果有人想看相关教程,可以留言。

我们主要来说说怎么定制错误处理。

通常来说,Laravel中的错误都是异常(Exception),我们都可以在app\Exceptions\handler.PHP中进行统一处理。Form Request确实也抛出了一个Illuminate\http\Exception\httpResponseException异常,但这个异常是在路由逻辑中就被特殊处理了。

首先我们来看看Form Request是如何被执行的:

Illuminate\ValIDation\ValIDationServiceProvIDer:

namespace Illuminate\Validation;

use Illuminate\Support\ServiceProvider;
use Illuminate\Contracts\Validation\ValidatesWhenResolved;

class ValidationServiceProvider extends ServiceProvider
{
/**

Register the service provider.@return void
*/
public function register()
{
$this->registerValIDationResolverHook(); // 看我看我看我
$this->registerPresenceVerif<a href="https://m.jb51.cc/tag/IE/" target="_blank" >IE</a>r();$this->registerVal<a href="https://m.jb51.cc/tag/ID/" target="_blank" >ID</a>ationFactory();

}

/**

Register the "ValIDatesWhenResolved" container hook.@return voID
*/
protected function registerValIDationResolverHook() // 对,就是我
{
// 这里可以看到对Val<a href="https://m.jb51.cc/tag/ID/" target="_blank" >ID</a>atesWhenRes<a href="https://m.jb51.cc/tag/ol/" target="_blank" >ol</a>ved的实现做了一个监听
$this->app->afterResolving(function (ValIDatesWhenResolved $resolved) {
$resolved->valIDate(); // 然后调用了它的val<a href="https://m.jb51.cc/tag/ID/" target="_blank" >ID</a>ate方法进行验证
});
}

// ...

你猜对了,Form Request就实现了这个Illuminate\Contracts\ValIDation\ValIDatesWhenResolved接口:

namespace Illuminate\Foundation\http;

use Illuminate\http\Request;
use Illuminate\http\Response;
use Illuminate\http\JsonResponse;
use Illuminate\Routing\Redirector;
use Illuminate\Container\Container;
use Illuminate\Contracts\ValIDation\ValIDator;
use Illuminate\http\Exception\httpResponseException;
use Illuminate\ValIDation\ValIDatesWhenResolvedTrait;
use Illuminate\Contracts\ValIDation\ValIDatesWhenResolved; // 是你
use Illuminate\Contracts\ValIDation\Factory as ValIDationFactory;

// 我们app\<a href="https://m.jb51.cc/tag/http/" target="_blank" >http</a>\Requests\Request便是继承于这个FormRequest
class FormRequest extends Request implements ValIDatesWhenResolved // 就是你
{
use ValIDatesWhenResolvedTrait; // 这个我们待会儿也要看看

// ...

FormRequest基类中的valIDate方法是由这个Illuminate\ValIDation\ValIDatesWhenResolvedTrait实现的:

Illuminate\ValIDation\ValIDatesWhenResolvedTrait:

namespace Illuminate\Validation;

use Illuminate\Contracts\Validation\ValidationException;
use Illuminate\Contracts\Validation\UnauthorizedException;

/**

Provides default implementation of ValidatesWhenResolved contract.
*/
trait ValidatesWhenResolvedTrait
{
/**

Validate the class instance.@return void
*/
public function validate() // 这里实现了validate方法
{
$instance = $this->getValIDatorInstance(); // 这里获取了Val<a href="https://m.jb51.cc/tag/ID/" target="_blank" >ID</a>ator实例

if (! $this->passesAuthorization()) {
$this->FailedAuthorization(); // 这是调用了访问授权的失败处理
} elseif (! $instance->passes()) {
$this->FailedValIDation($instance); // 这里调用了验证失败的处理,我们主要看这里
}
}

// ...

在valIDate里,如果验证失败了就会调用$this->FailedValIDation(),继续:

Illuminate\Foundation\http\FormRequest:

// ...

/**

Handle a failed validation attempt.@param \Illuminate\Contracts\Validation\Validator $validator@return mixed
*/
protected function failedValidation(Validator $validator)
{
throw new HttpResponseException($this->response( // 这里抛出了传说中的异常
$this->formatErrors($valIDator)
));
}

终于看到异常了!可是这个异常在另一个地方被处理了:

Illuminate\Routing\Route:

// ...

/**

Run the route action and return the response.@param \Illuminate\Http\Request $request@return mixed
*/
public function run(Request $request)
{
$this->container = $this->container ?: new Container;
try {  if (! is_string($this->action['uses'])) {    return $this-><a href="https://www.jb51.cc/tag/runc/" target="_blank" >runc</a>allable($request);  }  if ($this->custom<a href="https://www.jb51.cc/tag/dis/" target="_blank" >dis</a>patcherIsBound()) {    return $this->runWithCustom<a href="https://www.jb51.cc/tag/dis/" target="_blank" >dis</a>patcher($request);  }  return $this-><a href="https://www.jb51.cc/tag/runc/" target="_blank" >runc</a>ontr<a href="https://m.jb51.cc/tag/ol/" target="_blank" >ol</a>ler($request);} catch (<a href="https://m.jb51.cc/tag/http/" target="_blank" >http</a>ResponseException $e) { // 就是<a href="https://m.jb51.cc/tag/zheli/" target="_blank" >这里</a>  return $e->getResponse(); // <a href="https://m.jb51.cc/tag/zheli/" target="_blank" >这里</a>直接返回了Response给客户端}

}

// ...

至此,整个思路已然清晰,不过我们还是看看这里生成的httpResponseException异常中的Response是怎么生成的:

Illuminate\Foundation\http\FormRequest:

// ...

// 132行:
if ($this->AJAX() || $this->wantsJson()) { // 对AJAX请求的处理
return new JsonResponse($errors,422);
}

return $this->redirector->to($this->getRedirectUrl()) // 对普通表单提交的处理
->withinput($this->except($this->dontFlash))
->withErrors($errors,$this->errorBag);

// ...

相信你都看明白了。

如何实现自定义错误处理,这里提供两个思路,都需要重写app\http\Requests\Request的FailedValIDation:

抛出一个新异常,继承httpResponseException异常,重新实现getResponse方法,这个异常类我们可以放到app/Exceptions/下便于管理,错误返回依然交给Laravel;

抛出一个我们自定义的异常,在app\Exceptions\handler中处理。

具体实现这里就不写啦(参阅Laravel文档中关于错误处理部分,中文文档传送门),如果你有别的方法或者想法可以在评论中和我交流。

补充

如果你的Controller使用Illuminate\Foundation\ValIDation\ValIDatesRequests这个Trait的valIDate方法进行验证,同样的,这里验证失败也会抛出Illuminate\http\Exception\httpResponseException异常,可以参考上面的解决方案进行处理。

总结

以上是内存溢出为你收集整理的Laravel中使用FormRequest进行表单验证方法及问题汇总全部内容,希望文章能够帮你解决Laravel中使用FormRequest进行表单验证方法及问题汇总所遇到的程序开发问题。

如果觉得内存溢出网站内容还不错,欢迎将内存溢出网站推荐给程序员好友。

欢迎分享,转载请注明来源:内存溢出

原文地址: http://outofmemory.cn/langs/1269538.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2022-06-08
下一篇 2022-06-08

发表评论

登录后才能评论

评论列表(0条)

保存