PHP YII框架开发小技巧之模型(models)中rules自定义验证规则 |
|
YII的models中的rules部分是一些表单的验证规则,对于表单验证十分有用,在相应的视图(views)里面添加了表单,在表单被提交之前程序都会自动先来这里面的规则里验证,只有通过对其有效的限制规则后才能被提交,可以很有效地保证表单安全和信息的有效性 。还是给大家具体说明一下: 以下是视图(views)部分的简单代码:
<?php $form=$this->beginWidget(CActiveForm, array(
id=>tag-form,
enableAjaxValidation=>false,
)); ?>
<div class="row">
<?php echo $form->labelEx($model,tagname); ?>
<?php echo $form->textField($model,tagname,array(size=>20,maxlength=>32)); ?>
</div>
<div class="row">
<?php echo $form->labelEx($model,tagtype); ?>
<?php echo $form->radioButtonList($model,tagtypearray(1=>"普通TAG",2=>"系统默认TAG"),array(separator=>,labelOptions=>array(class=>tagtypelabel))); ?>
</div>
<?php echo $form->errorSummary($model); ?>
<div class="row buttons">
<?php echo CHtml::submitButton($model->isNewRecord ? 添加 : 修改); ?>
</div>
<?php $this->endWidget(); ?>
模型(models)中rules部分的简单代码:
public function rules()
{
return array(
array(tagname,tagtype, required),
array(tagtype, numerical, integerOnly=>true),
array(tagname, length, max=>32),
array(tagname, match, pattern=>/^[\x{4e00}-\x{9fa5}A-Za-z0-9]+$/u,
message=>标签不合法,必须为汉字、字母或者数字!),
array(tagname, checktagname, on=>create,update),//插入TAG时检查是否已经存在该tag
array(tagid, tagname, tagtype, safe, on=>search),
);
}
系统默认有这些验证规则: boolean : CBooleanValidator 的别名, 确保属性的值是CBooleanValidator::trueValue 或 CBooleanValidator::falseValue . 基本上还是比较全面的,一般的都够用了,但是还是有时候有的验证需要自定义 。就以上面的代码为例,我们在添加TAG时需要检查系统之前是否已经存在这个TAG,如果存在则不让用户添加 。这个就需要在添加之前去查询数据库,看该TAG是否已经存在,这里我们就需要自定一个验证规则了 。 关键有一下两个步骤: 1、在rules中 添加代码:array(tagname, checktagname, on=>create,update),//插入TAG时检查是否已经存在该tag 注:我在其中用了 on=>create,update,所以这个验证规则之对create,update场景生效 2、在该模型(models)中添加验证函数:
public function checktagname($attribute,$params){
$oldtag = Tag::model()->findByAttributes(array(tagname=>$this->tagname));
if($oldtag->tagid > 0){
$this->addError($attribute, 该TAG已经存在!);
}
}
其中需要说明的是: (1)该验证函数的参数必须是($attribute,$params),不能缺少其中任何一个; (2)$this->addError($attribute, 该TAG已经存在!);这个是你想要在视图中输出的错误提示信息 。 就是这么简单,有了这个方法,表单验证的各种想要的规则就都可以自定义了 。 下面给大家介绍Yii自定义验证规则 最简单的定义验证规则的方法是在使用它的模型(model)内部定义 。 比方说,你要检查用户的密码是否足够安全. 通常情况下你会使用 CRegularExpression 方法验证,但为了本指南,我们假设不存在此验证方法. 首先在模型(model)中添加两个常量 const WEAK = 0;
/**
* @return array validation rules for model attributes.
*/
public function rules()
{
return array(
array(password, passwordStrength, strength=>self::STRONG),
);
}
确保你写的规则不是一个已经存在的规则,否则将会报错. 现在要做的是在模型(model)中创建一个名称为上面填写的规则的方法(即 passwordStrength) 。
/**
* check if the user password is strong enough
* check the password against the pattern requested
* by the strength parameter
* This is the passwordStrength validator as declared in rules().
*/
public function passwordStrength($attribute,$params)
{
if ($params[strength] === self::WEAK)
$pattern = /^(?=.*[a-zA-Z0-9]).{5,}$/;
elseif ($params[strength] === self::STRONG)
$pattern = /^(?=.*\d(?=.*\d))(?=.*[a-zA-Z](?=.*[a-zA-Z])).{5,}$/;
if(!preg_match($pattern, $this->$attribute))
$this->addError($attribute, your password is not strong enough!);
}
刚才创建的方法需要两个参数:* $attribute 需要验证的属性* $params 在规则中自定义的参数 在模型的 rules 方法中我们验证的是 password 属性,所以在验证规则中需要验证的属性值应该是 password. 在 rules 方法中我们还设置了自定义的参数 strength,它的值将会放到 $params 数组中. 你会发现在方法中我们使用了 CModel::addError(). 添加错误接受两个参数:第一个参数是在表单中显示错误的属性名,第二个参数时显示的错误信息 。 完整的方法:继承 CValidator 类 如果你想把规则使用在多个模型(model)中,最好的方法时继承 CValidator 类 。 继承这个类你可以使用像 CActiveForm::$enableClientValidation (Yii 1.1.7 版本后可用) 类似的其他功能 。 创建类文件 首先要做的是创建类文件.最好的方法时类的文件名和类名相同,可以使用 yii 的延迟加载(lazy loading)功能 。 让我们在应用(application)的扩展(extensiions)目录(在 protected 文件夹下)下新建一个文件夹. 将目录命名为: MyValidators 然后创建文件: passwordStrength.php 在文件中创建我们的验证方法
class passwordStrength extends CValidator
{
public $strength;
private $weak_pattern = /^(?=.*[a-zA-Z0-9]).{5,}$/;
private $strong_pattern = /^(?=.*\d(?=.*\d))(?=.*[a-zA-Z](?=.*[a-zA-Z])).{5,}$/;
...
}
在类中创建属性,此属性为在验证规则中使用的参数. CValidator 会自动根据参数来填充这些属性. 我们也创建了两个其他的属性,它们为 preg_match 函数使用的正则表达式. 现在我们应该重写父类的抽象方法(abstract method) validateAttribute
/**
* Validates the attribute of the object.
* If there is any error, the error message is added to the object.
* @param CModel $object the object being validated
* @param string $attribute the attribute being validated
*/
protected function validateAttribute($object,$attribute)
{
// check the strength parameter used in the validation rule of our model
if ($this->strength == weak)
$pattern = $this->weak_pattern;
elseif ($this->strength == strong)
$pattern = $this->strong_pattern;
// extract the attribute value from its model object
$value=$object->$attribute;
if(!preg_match($pattern, $value))
{
$this->addError($object,$attribute,your password is too weak!);
}
}
上面的方法我认为就不用解释了.当然你也可以在 if 的条件中使用常量,我推荐使用. |