關(guān)于Laravel參數(shù)驗證的一些疑與惑
驗證器怎么創(chuàng)建的,誰創(chuàng)建的
Laravel 文檔調(diào)用驗證器,除了通過控制器,還有就是通過Facades的方式創(chuàng)建驗證器對象。Validator::make($data,$rule,$message)。
config/app.php 中注冊了'Validator' => Illuminate\Support\Facades\Validator::class。
<?php
namespace Illuminate\Support\Facades;
/**
* @see \Illuminate\Validation\Factory
*/
class Validator extends Facade
{
/**
* Get the registered name of the component.
*
* @return string
*/
protected static function getFacadeAccessor()
{
return 'validator';
}
}
從上面可以看出,Validator的實際實現(xiàn)類是容器中的validator對象,那這個validator對象是哪個?
<?php
namespace Illuminate\Foundation;
...
class Application extends Container implements ApplicationContract, HttpKernelInterface
{
...
public function registerCoreContainerAliases()
{
foreach ([
...
'validator'=> [
\Illuminate\Validation\Factory::class,
\Illuminate\Contracts\Validation\Factory::class
],
])
...
}
...
}
可以看出,最終創(chuàng)建驗證器是通過實現(xiàn)\Illuminate\Contracts\Validation\Factory接口的\Illuminate\Validation\Factory類創(chuàng)建的。再來看看,這個工廠類怎么創(chuàng)建實際的驗證器的。
//\Illuminate\Contracts\Validation\Factory 源碼
protected function resolve(array $data, array $rules, array $messages, array $customAttributes)
{
if (is_null($this->resolver)) {
return new Validator(
$this->translator,
$data,
$rules,
$messages,
$customAttributes
);
}
return call_user_func(
$this->resolver,
$this->translator,
$data,
$rules,
$messages,
$customAttributes
);
}
到這里,可以看出Laravel的驗證器的創(chuàng)建都是通過特定的工廠類創(chuàng)建。
如果需要自定義驗證器類(比如我需要把5.8的一些新功能遷移到5.5的版本上),有兩種方式:
一,創(chuàng)建一個自定義的工廠類。然后在AppServiceProvider中重新綁定新的驗證器工廠創(chuàng)建類;
二,AppServiceProvider中通過resolver方法設(shè)置工廠類的resolver屬性,接管驗證器的實例化,例如:
Validator::resolver(function($translator, $data, $rules, $messages, $customAttributes){
return new ExtendValidator($translator, $data, $rules, $messages, $customAttributes);
});
如何自定義驗證規(guī)則
Laravel本身提供了很多通用的參數(shù)驗證規(guī)則,但是對于一些特定的場景,還是需要提供驗證規(guī)則的擴(kuò)展。
Laravel驗證規(guī)則的擴(kuò)展有兩種方式。
1 通過extend方法擴(kuò)展
//這是一個簡單的參數(shù)比較的驗證規(guī)則,Laravel5.8中提供,Laravel5.5中未提供
//驗證規(guī)則如下: 'max_num'=>'gte:min',
Validator::extend('gte',function($attribute, $value, $parameters, $validator){
if($value>=data_get($validator->getData(),$parameters[0]))
{
return true;
}
return false;
});
//\Illuminate\Contracts\Validation\Factory 源碼
public function extend($rule, $extension, $message = null)
{
$this->extensions[$rule] = $extension;
if ($message) {
$this->fallbackMessages[Str::snake($rule)] = $message;
}
}
//\Illuminate\Validation\Validator 源碼
protected function callExtension($rule, $parameters)
{
$callback = $this->extensions[$rule];
if (is_callable($callback)) {
return call_user_func_array($callback, $parameters);
} elseif (is_string($callback)) {
return $this->callClassBasedExtension($callback, $parameters);
}
}
protected function validateAttribute($attribute, $rule)
{
...
$method = "validate{$rule}";
if ($validatable && ! $this->$method($attribute, $value, $parameters, $this)) {
$this->addFailure($attribute, $rule, $parameters);
}
}
public function __call($method, $parameters)
{
$rule = Str::snake(substr($method, 8));
if (isset($this->extensions[$rule])) {
return $this->callExtension($rule, $parameters);
}
throw new BadMethodCallException(sprintf(
'Method %s::%s does not exist.', static::class, $method
));
}
Factory提供了extend方法用于擴(kuò)展規(guī)則驗證方法。所有的擴(kuò)展規(guī)則最終都會被傳到驗證器中。驗證器在驗證參數(shù)的過程中,如果找到匹配的驗證規(guī)則,則直接進(jìn)行驗證。否則調(diào)用魔術(shù)方法__call查找擴(kuò)展驗證函數(shù)。擴(kuò)展函數(shù)返回布爾值,返回true則表示驗證通過,返回false表示驗證失敗。
2 通過自定義規(guī)則類擴(kuò)展
Laravel 中提供了Illuminate\Contracts\Validation\Rule接口,只有實現(xiàn)了這個接口的類都認(rèn)為是符合的自定義驗證規(guī)則類。
<?php
namespace Illuminate\Contracts\Validation;
interface Rule
{
/**
* Determine if the validation rule passes.
*
* @param string $attribute
* @param mixed $value
* @return bool
*/
public function passes($attribute, $value);
/**
* Get the validation error message.
*
* @return string
*/
public function message();
}
自定義規(guī)則類需要實現(xiàn)的方法有passes方法,用于驗證參數(shù)是否合法。message方法,用于提供驗證失敗的錯誤提示信息。
使用自定義驗證類,相對于extend方法擴(kuò)展有一個很大的bug就是無法在自定義類中獲取到當(dāng)期的驗證器對象。從而導(dǎo)致在當(dāng)前擴(kuò)展的驗證規(guī)則中,只能過獲取到需要驗證的數(shù)據(jù),而獲取不到其他的字段數(shù)據(jù),無法進(jìn)行聯(lián)合字段的驗證。像上面比較兩個字段的大小的驗證規(guī)則就無法實現(xiàn)。
如果想要通過自定義驗證規(guī)則類實現(xiàn)上面兩個字段大小比較的驗證規(guī)則,則需要自定義驗證類,修改validateUsingCustomRule方法,將當(dāng)期驗證器傳入到自定義驗證規(guī)則實例對象中去。
protected function validateUsingCustomRule($attribute, $value, $rule)
{
if(method_exists($rule, 'setValidator'))
{
$rule->setValidator($this);
}
return parent::validateUsingCustomRule($attribute,$value,$rule);
}
如何實現(xiàn)用當(dāng)期類方法作為驗證規(guī)則驗證函數(shù)
像Yii2中,因為基本上所有的對象都有驗證方法,所以很容易用當(dāng)期類方法作為驗證規(guī)則驗證函數(shù)。
例如,一個驗證規(guī)則如下,表示用當(dāng)期類的validateMinNum對參數(shù)進(jìn)行驗證,那么,這樣的一個功能,如何在Laravel中實現(xiàn)呢。
['min_num'=>'validateMinNum']
方法1 通過自定義類實現(xiàn) Laravel提供了ClosureValidationRule自定義驗證類,用來添加回調(diào)函數(shù)的驗證。
例如
$rule = [ 'min'=>new ClosureValidationRule([$this,'checkv']) ]; $data = ['min'=>10]; $v = Validator::make($data,$rule);
方法2 通過extend方式實現(xiàn)
$rule = [
'min'=>'checkv'
];
Validator::extend('checkv',[$this,'checkv']);
但是這種方式對驗證器的影響是全局的。不建議使用。
總結(jié)
通過以上源碼的學(xué)習(xí),可以看出Laravel驗證器的創(chuàng)建都是用過驗證器工廠類創(chuàng)建的。如果需要自定義驗證器,可以通過修改驗證器工廠類,或者設(shè)置驗證器工廠類的resolver屬性接管驗證器的實例化。
驗證規(guī)則的擴(kuò)展有兩種方式,一種是通過extend方式實現(xiàn)。extend方式對驗證器的影響是全局的,整個運行進(jìn)程有效??梢垣@取到驗證器本身,因此可以做多個字段關(guān)系的驗證;另一種是通過自定義規(guī)則類實現(xiàn)。自定義規(guī)則了只對使用自定義規(guī)則類的驗證有效。但是自定義規(guī)則類本身無法直接獲取到驗證器本身,不能夠做多個字段關(guān)系的驗證。如果需要實現(xiàn),則需要使用自定義驗證器,將驗證器傳入到驗證規(guī)則中去。
Laravel本身提供了ClosureValidationRule的驗證規(guī)則用于處理回調(diào)函數(shù)驗證規(guī)則。同時也可以使用extend方式進(jìn)行回調(diào)函數(shù)的驗證。
好了,以上就是這篇文章的全部內(nèi)容了,希望本文的內(nèi)容對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,謝謝大家對腳本之家的支持。
- Laravel框架表單驗證詳解
- Laravel中使用FormRequest進(jìn)行表單驗證方法及問題匯總
- 在Laravel框架里實現(xiàn)發(fā)送郵件實例(郵箱驗證)
- Laravel框架實現(xiàn)發(fā)送短信驗證功能代碼
- Laravel 5框架學(xué)習(xí)之表單驗證
- Laravel4中的Validator驗證擴(kuò)展用法詳解
- 使用 laravel sms 構(gòu)建短信驗證碼發(fā)送校驗功能
- Laravel中unique和exists驗證規(guī)則的優(yōu)化詳解
- Laravel框架用戶登陸身份驗證實現(xiàn)方法詳解
- Laravel5.2使用Captcha生成驗證碼實現(xiàn)登錄(session巨坑)
相關(guān)文章
PHP SPL標(biāo)準(zhǔn)庫之文件操作(SplFileInfo和SplFileObject)實例
這篇文章主要介紹了PHP SPL標(biāo)準(zhǔn)庫之文件操作(SplFileInfo和SplFileObject)實例,本文講解SplFileInfo用來獲取文件詳細(xì)信息、SplFileObject遍歷、查找指定行、寫入csv文件等內(nèi)容,需要的朋友可以參考下2015-05-05
php檢測網(wǎng)頁是否被百度收錄的函數(shù)代碼
下面給出一段php函數(shù),功能是檢測一個網(wǎng)頁是否被百度收錄,直接給出代碼2013-10-10
詳解Grunt插件之LiveReload實現(xiàn)頁面自動刷新(兩種方案)
這篇文章主要通過兩種方案詳解Grunt插件之LiveReload實現(xiàn)頁面自動刷新,需要的朋友可以參考下2015-07-07
php使用Session和文件統(tǒng)計在線人數(shù)
這篇文章主要介紹了php使用Session和文件統(tǒng)計在線人數(shù),本文直接給出實例代碼,需要的朋友可以參考下2015-07-07
laravel框架分組控制器和分組路由實現(xiàn)方法示例
這篇文章主要介紹了laravel框架分組控制器和分組路由實現(xiàn)方法,結(jié)合實例形式分析了laravel框架分組控制器和分組路由的基本定義與使用方法,需要的朋友可以參考下2020-01-01
淺談thinkphp的nginx配置,以及重寫隱藏index.php入口文件方法
今天小編就為大家分享一篇淺談thinkphp的nginx配置,以及重寫隱藏index.php入口文件方法,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧2019-10-10

