模型类DynamicModel主要用于实现模型内的数据验证yii2\base\DynamicModel.php

 <?php
/**
* @link http://www.yiiframework.com/
* @copyright Copyright (c) 2008 Yii Software LLC
* @license http://www.yiiframework.com/license/
*/
namespace yii\base; use yii\validators\Validator; /**
* DynamicModel is a model class primarily used to support ad hoc data validation.
* DynamicModel是一种主要用于支持ad hoc数据验证模型类
* The typical usage of DynamicModel is as follows,
*
* ```php
* public function actionSearch($name, $email)
* {
* $model = DynamicModel::validateData(compact('name', 'email'), [
* [['name', 'email'], 'string', 'max' => 128],
* ['email', 'email'],
* ]);
* if ($model->hasErrors()) {
* // validation fails
* } else {
* // validation succeeds
* }
* }
* ```
*
* The above example shows how to validate `$name` and `$email` with the help of DynamicModel.
* 上面的例子演示了如何用DynamicModel验证用户名`$name`和邮箱`$email`
* The [[validateData()]] method creates an instance of DynamicModel, defines the attributes
* using the given data (`name` and `email` in this example), and then calls [[Model::validate()]].
* validateData() 方法会创建一个 DynamicModel 的实例对象。通过给定数据定义模型特性,之后调用Model::validate() 方法。
* You can check the validation result by [[hasErrors()]], like you do with a normal model.
* You may also access the dynamic attributes defined through the model instance, e.g.,
* 可以通过[[hasErrors()]]方法获取验证结果
* `$model->name` and `$model->email`.
*
* Alternatively, you may use the following more "classic" syntax to perform ad-hoc data validation:
* 除此之外,你也可以用如下的更加“classic(传统)”的语法来执行临时数据验
* ```php
* $model = new DynamicModel(compact('name', 'email'));
* $model->addRule(['name', 'email'], 'string', ['max' => 128])
* ->addRule('email', 'email')
* ->validate();
* ```
*
* DynamicModel implements the above ad-hoc data validation feature by supporting the so-called
* "dynamic attributes". It basically allows an attribute to be defined dynamically through its constructor
* or [[defineAttribute()]].
* 实现了上述特殊数据模型验证功能支持的“动态属性”。允许通过它的构造函数或 [[defineAttribute()]]来定义一个属性
* @author Qiang Xue <qiang.xue@gmail.com>
* @since 2.0
*/
class DynamicModel extends Model
{
private $_attributes = [];//动态模型内动态属性 /**
* Constructors.构造函数,用于将传入的属性赋值给_attributes,便于使用
* @param array $attributes the dynamic attributes (name-value pairs, or names) being defined被定义的动态属性
* @param array $config the configuration array to be applied to this object.用于该对象的配置数组。
*/
public function __construct(array $attributes = [], $config = [])
{
foreach ($attributes as $name => $value) {//遍历传入的属性
if (is_integer($name)) {//如果是整型,说明只传入了属性名,将属性名写入_attributes
$this->_attributes[$value] = null;
} else {
$this->_attributes[$name] = $value;//按键值对的形式写入
}
}
parent::__construct($config);//调用父类的配置
} /**
* @inheritdoc 重写__get方法,从_attributes中取值
*/
public function __get($name)
{
if (array_key_exists($name, $this->_attributes)) {
//如果传入的$name在数组_attributes中存在,则从_attributes中取值
return $this->_attributes[$name];
} else {//否则调用父类的__get方法取属性值
return parent::__get($name);
}
} /**
* @inheritdoc 重写__set方法,给_attributes设置值
*/
public function __set($name, $value)
{
if (array_key_exists($name, $this->_attributes)) {
//如果传入的$name在数组_attributes中存在,则将动态属性$name的值设置为$value
$this->_attributes[$name] = $value;
} else {
parent::__set($name, $value);//调用父类的__set方法设置属性值
}
} /**
* @inheritdoc 同上 重写__isset方法,判断_attributes中是否设置$name值
*/
public function __isset($name)
{
if (array_key_exists($name, $this->_attributes)) {
return isset($this->_attributes[$name]);
} else {
return parent::__isset($name);
}
} /**
* @inheritdoc 同上,重写__unset方法,删除_attributes中的$name属性值
*/
public function __unset($name)
{
if (array_key_exists($name, $this->_attributes)) {
unset($this->_attributes[$name]);
} else {
parent::__unset($name);
}
} /**
* Defines an attribute. 定义动态属性的方法
* @param string $name the attribute name 属性名
* @param mixed $value the attribute value 属性值
*/
public function defineAttribute($name, $value = null)
{
$this->_attributes[$name] = $value;
} /**
* Undefines an attribute. 用于删除动态属性的方法
* @param string $name the attribute name 属性名
*/
public function undefineAttribute($name)
{
unset($this->_attributes[$name]);
} /**
* Adds a validation rule to this model. 添加验证规则
* You can also directly manipulate [[validators]] to add or remove validation rules.
* This method provides a shortcut.
* 可以直接调用[[validators]]来添加或者删除验证规则,本方法提供了一个短方法
* @param string|array $attributes the attribute(s) to be validated by the rule 进行验证的属性
* @param mixed $validator the validator for the rule.This can be a built-in validator name,
* a method name of the model class, an anonymous function, or a validator class name.
* 规则的验证。这是一个内置验证器的名字, 一个模型类的方法名,一个匿名函数或一个验证器类的名称。
* @param array $options the options (name-value pairs) to be applied to the validator
* (name-value)被应用到验证器
* @return static the model itself 模型本身
*/
public function addRule($attributes, $validator, $options = [])
{
$validators = $this->getValidators();//所有的验证规则对象
//生成Validator对象,并且插入 $validators中
$validators->append(Validator::createValidator($validator, $this, (array) $attributes, $options)); return $this;
} /**
* Validates the given data with the specified validation rules.通过指定的规则验证给定的数据
* This method will create a DynamicModel instance, populate it with the data to be validated,
* create the specified validation rules, and then validate the data using these rules.
* @param array $data the data (name-value pairs) to be validated
* @param array $rules the validation rules. Please refer to [[Model::rules()]] on the format of this parameter.
* @return static the model instance that contains the data being validated
* @throws InvalidConfigException if a validation rule is not specified correctly.
*/
public static function validateData(array $data, $rules = [])
{
/* @var $model DynamicModel */
$model = new static($data);//实例化调用类,将$data赋值给_attributes
if (!empty($rules)) {
$validators = $model->getValidators();//获取所有定义的验证规则
foreach ($rules as $rule) {
if ($rule instanceof Validator) {
$validators->append($rule);//如果$rule是Validator的实例,则添加到$validators中
} elseif (is_array($rule) && isset($rule[], $rule[])) { // attributes, validator type
//如果$rule是数组,则判断动态属性和验证类型是否存在,创建Validator对象,添加到$validators中
$validator = Validator::createValidator($rule[], $model, (array) $rule[], array_slice($rule, ));
$validators->append($validator);
} else {//抛出异常
throw new InvalidConfigException('Invalid validation rule: a rule must specify both attribute names and validator type.');
}
}
} $model->validate();//执行验证 return $model;
} /**
* @inheritdoc 返回所有的动态属性
*/
public function attributes()
{
return array_keys($this->_attributes);
}
}

yii2源码学习笔记(十三)的更多相关文章

  1. yii2源码学习笔记(九)

    Application是所有应用程序类的基类,接下来了解一下它的源码.yii2\base\Application.php. <?php /** * @link http://www.yiifra ...

  2. yii2源码学习笔记(八)

    Action是所有控制器的基类,接下来了解一下它的源码.yii2\base\Action.php <?php /** * @link http://www.yiiframework.com/ * ...

  3. 老刘 Yii2 源码学习笔记之 Action 类

    Action 的概述 InlineAction 就是内联动作,所谓的内联动作就是放到controller 里面的 actionXXX 这种 Action.customAction 就是独立动作,就是直 ...

  4. yii2源码学习笔记(二十)

    Widget类是所有部件的基类.yii2\base\Widget.php <?php /** * @link http://www.yiiframework.com/ * @copyright ...

  5. yii2源码学习笔记(十八)

    View继承了component,用于渲染视图文件:yii2\base\View.php <?php /** * @link http://www.yiiframework.com/ * @co ...

  6. yii2源码学习笔记(十七)

    Theme 类,应用的主题,通过替换路径实现主题的应用,方法为获取根路径和根链接:yii2\base\Theme.php <?php /** * @link http://www.yiifram ...

  7. yii2源码学习笔记(十四)

    Module类是模块和应用类的基类. yiisoft\yii2\base\Module.php <?php /** * @link http://www.yiiframework.com/ * ...

  8. yii2源码学习笔记(十一)

    Controller控制器类,是所有控制器的基类,用于调用模型和布局. <?php /** * @link http://www.yiiframework.com/ * @copyright C ...

  9. yii2源码学习笔记(六)

    Behvaior类,Behavior类是所有事件类的基类: 目录yii2\base\Behavior.php <?php /** * @link http://www.yiiframework. ...

随机推荐

  1. Rediss_基本介绍

    Redis是典型的NoSQL数据库( Not Only SQL) NoSQL数据库: NoSQL,泛指非关系型的数据库.随着互联网web2.0网站的兴起, 传统的关系数据库在应付web2.0网站, 特 ...

  2. Java-Web监听器

    在WEB端实现监听实质: 实现一系列的监听接口(实现相应的接口,覆写各接口中相应的方法,在相应的事件触发的时候会执行自己的监听器中的覆写的方法,在各个方法中完成自己想要的操作,从而实现了监听) 监听- ...

  3. python学习(4)

    python(4)4.1 高阶函数:map/reduce map:实际上map也是一个函数,只不过他可以在参数里面包含别的函数.他有两种参数:第一种:函数(作用在后面要说的序列上),第二种:一个序列  ...

  4. SqlServer数据库的一些方法的用途

    一直分不清这三种方法的具体用法现在终于找齐了 ExecuteNonQuery方法和ExecuteScalar方法和ExecuteReader方法的区别 (1)ExecuteNonQuery():执行命 ...

  5. Ubuntu下开启root登陆

    亲手安装过Ubuntu的童鞋都知道,默认安装只会添加一个普通用户名和密码,而超级用户权限则是利用sudo命令来执行.在Ubuntu下使用root登陆或者在shell中用su命令切换到root时会提示错 ...

  6. IOS开发UI篇之──自定义UIActionSheet

    转载自:http://www.cnblogs.com/pengyingh/articles/2343200.html UIActionSheet类系IOS开发中实现警告框的重要的类,而在好多应用中,都 ...

  7. TransactionScope 对该事务的状态无效 和一些注意事项

    使用TransactionScope 的时候要操作同一种数据库操作方式,不能一个方法用ado.net ,另外一个方法用EF,那样会报 "该事务管理器已经禁止了它对远程/网络事务的支持&quo ...

  8. 在IT在系统中使用多租户技术的跨部门和虚拟团队的解决方案为员工提供(草案)

    1 前言 经过多年的企业信息化建设,Office系统逐步形成有9营业场所的分部门.9专业应用子系统.20独立的信息模块.330一种方法.这些系统或模块内置于Microsoft IIS.Apache T ...

  9. 谋哥:App排行榜的秘密

    App在改变世界,改变人们的生活.       如今购物大家都用淘宝.京东,吃饭你会用饭否,看天气预报你用墨迹天气,看视频用优酷.K歌你用唱吧,聊天联系你用微信,看新闻你用今日头条等等.你的生活由你自 ...

  10. 系统调用服务号 linux 2.6.32

    http://www.cnblogs.com/xcywt/p/4998963.html 系统定义符号集中声明在 /usr/src/kernels/linux-2.6.32/include/linux/ ...