YII框架分析笔记3:表单模型和验证

表单模型CFormModel绝大部分继承CModelCModel,由于表模型数据不需要持久化,所以主要在验证操作上。下面以框架脚手架生成的网站登录为例说明表单模型。

  1. //模型中的验证规则  
  2. public function rules()  
  3. {  
  4.     return array(  
  5.         array('username, password', 'required'),  
  6.         array('rememberMe', 'boolean'),  
  7.         array('password', 'authenticate'),  
  8.     );  
  9. }  
  1. $model=new LoginForm;//创建一个登录表单模型  
  2. $model->attributes=$_POST['LoginForm'];//传入登录数据  
  3. CModel->setAttributes($values,$safeOnly=true)//调用CModel的setter方法  
  4.   
  5. //返回model中relues中定义的安全数据,经过下面的调用流程获取一个验证器集合  
  6. //每个验证器由CValidator::createValidator($rule[1],$this,$rule[0],array_slice($rule,2))  
  7. CModel->getValidators()  
  8. ->createValidators()  
  9. ->rules()  
  10. CModel->validate();//遍历验证器执行验证  

验证器内部实现
表单模型核心部分在验证上,下面探讨其实现方式。
YII框架中验证以独立组件集合的形式存在,CValidator是所有验证器的基类。还是以登录验证为例。rules方法中包含required、boolean、authenticate三个验证器

  1. public static function createValidator($name,$object,$attributes,$params=array())  
  2. {  
  3.     if(is_string($attributes))  
  4.         $attributes=preg_split('/[\s,]+/',$attributes,-1,PREG_SPLIT_NO_EMPTY);  
  5.   
  6.     if(isset($params['on']))  
  7.     {  
  8.         if(is_array($params['on']))  
  9.             $on=$params['on'];  
  10.         else  
  11.             $on=preg_split('/[\s,]+/',$params['on'],-1,PREG_SPLIT_NO_EMPTY);  
  12.     }  
  13.     else  
  14.         $on=array();  
  15.   
  16.     if(method_exists($object,$name))  
  17.     {  
  18.         $validator=new CInlineValidator;  
  19.         $validator->attributes=$attributes;  
  20.         $validator->method=$name;  
  21.         if(isset($params['clientValidate']))  
  22.         {  
  23.             $validator->clientValidate=$params['clientValidate'];  
  24.             unset($params['clientValidate']);  
  25.         }  
  26.         $validator->params=$params;  
  27.         if(isset($params['skipOnError']))  
  28.             $validator->skipOnError=$params['skipOnError'];  
  29.     }  
  30.     else  
  31.     {  
  32.         $params['attributes']=$attributes;  
  33.         if(isset(self::$builtInValidators[$name]))  
  34.             $className=Yii::import(self::$builtInValidators[$name],true);  
  35.         else  
  36.             $className=Yii::import($name,true);  
  37.         $validator=new $className;  
  38.         foreach($params as $name=>$value)  
  39.             $validator->$name=$value;  
  40.     }  
  41.   
  42.     $validator->on=empty($on) ? array() : array_combine($on,$on);  
  43.   
  44.     return $validator;  
  45. }  

创建验证器流程
1、获取属性(array $attributes)和使用场景(array $on)
2、判断验证器是否存在model中,如果存在创建一个内联验证器CInlineValidator,如果不存在,执行第3步
3、如果验证器是框架自带的导入自带验证器,否则导入外部验证器,然后实例化并赋值。
当触发validate()时候,可能产生的错误将被存储在model中,可以通过调用 CModel::getErrors() 和CModel::getError() 提取这些错误信息。

posted @ 2015-09-21 19:57  SunsCheung  阅读(217)  评论(0编辑  收藏  举报