热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

PHP开发框架YiiFramework教程(8)使用FormModel

PHP开发框架YiiFramework教程(8)使用FormModel

通过前面的学习,我们了解了Yii Web应用的基本组成部分,也会编写像Hangman猜单词游戏这样简单的应用。在第一个例子 Yii Framework 开发简明教程(1) 第一个应用Hello World 我们介绍了Yii Web应用采用MVC模型,也说明了本教程目的是通过不 同的视角(主要是通过开发Windows应用C++,C#程序员的角度)帮助Windows 桌面应用或ASP.Net程序员较快的掌握PHP Yii Framework应用程序框架。

前面我们介绍了通过CHtml创建View(页面视图Form),通过CController来处理用户提交事件 ,和Windows 桌面应用或ASP.Net做个类比, Yii 中视图View (HTML Form) 类似于WinForm或是Asp.Net 的Page。 控制类 Controller类似Windows桌面应用或Asp.Net的事件处理(Code-Behind)类。不同的是Asp.Net和Windows 桌面应用可以为UI中各 个UI组件,比如文本框,按钮定义Id,然后为不同的UI组件添加事件处理。PHP应用或是Yii应用没有对应的机制可以为定义在 HTML Form中的UI组件定义一个Id,并为UI组件定义事件处理。 然而Yii 框架提供了CFormModel 可以支持类似的功能,简单的 说,通过CFormModel,可以为HTML Form 中的UI小组件定义变量,并且可以在其控制类Controller中访问这些变量。每个Yii View(Form)一般都提供一个“提交”按钮(Submit Button),用户点击这个“提交按钮”触发CController对象对应的 actionXXX 方法,在actionXXX 方法中可以通过CFormModel来访问HTML Form的UI组件的值。

前面教程中说过Yii中的模 型(Model)是 CModel 或其子类的实例。模型用于保持数据以及与其相关的业务逻辑,

Yii 实现了两种类型的模型:表 单模型和 Active Record。二者均继承于相同的基类 CModel。

表单模型是 CFormModel 的实例。表单模型用于保持从用 户的输入获取的数据。 这些数据经常被获取,使用,然后丢弃。例如,在一个登录页面中, 我们可以使用表单模型用于表示由 最终用户提供的用户名和密码信息。更多详情,请参考 使用表单。本篇介绍CFormModel的用法,

Active Record (AR) 是一种用于通过面向对象的风格抽象化数据库访问的设计模式。 每个 AR 对象是一个CActiveRecord 或其子类的实例。代表数 据表中的一行。 行中的字段对应 AR 对象中的属性。更多关于 AR 的细节请阅读 Active Record. 后面介绍数据库使用时再介 绍。

本篇使用一个简单的登录界面来介绍FormModel的用法,本例下载。

1. 定义模型类

下面我们创建了一个 LoginForm (protected/models/LoginForm.php) 模型类用于在一个登录页中收集用户的输入。 由于登 录信息只被用于验证用户,并不需要保存,因此我们将 LoginForm 创建为一个 表单模型。

class LoginForm extends

CFormModel    
{    
    public $username;    
    public $password;    
    public $rememberMe=false;    
}2. 声明验证规则

一旦用户提交了他的输入,模型被填充,我们就需要在使用前确保用户的输入是有效的。 这是通过将用户的输入和一系列规则执行验证实现的。我们在 rules() 方法中指定这些验证规则, 此方法应返回一个规则配置 数组。

 

class LoginForm extends CFormModel    
{    
    public $username;    
    public $password;    
    public $rememberMe=false;    
        
    private $_identity;    
        
    public function rules()    
    {    
        return array(    
            array('username, password', 'required'),    
            array('rememberMe', 'boolean'),    
            array('password', 'authenticate'),    
        );    
    }    
        
    public function authenticate($attribute,$params)    
    {    
        $this->_identity=new UserIdentity($this->username,    
              $this->password);    
        if(!$this->_identity->authenticate())    
            $this->addError('password','错误的用户名或密码。');    
    }    
}上述代码指定:username 和 password 为必填项, password 应被验证(authenticated),rememberMe 应该是一 个布尔值。

rules() 返回的每个规则必须是以下格式:

array('AttributeList', 'Validator',

'on'=>'ScenarioList', ...附加选项)

其中 AttributeList(特性列表) 是需要通过此规则验证的特性列表字符 串,每个特性名字由逗号分隔;Validator(验证器) 指定要执行验证的种类;on 参数是可选的,它指定此规则应被应用到的场 景列表; 附加选项是一个名值对数组,用于初始化相应验证器的属性值。

有三种方式可在验证规则中指定 Validator 。第一, Validator 可以是模型类中一个方法的名字,就像上面示例中的 authenticate 。验证方法必须是下面的结构:

/**   
 * @param string 所要验证的特性的名字   
 * @param array 验证规则中指定的选项   
 */
public function ValidatorName($attribute,$params) { ... }第二,Validator 可以是一个验证器类的名字,当此 规则被应用时, 一个验证器类的实例将被创建以执行实际验证。规则中的附加选项用于初始化实例的属性值。 验证器类必须继 承自 CValidator。

第三,Validator 可以是一个预定义的验证器类的别名。在上面的例子中, required 名字是 CRequiredValidator 的别名,它用于确保所验证的特性值不为空。 下面是预定义的验证器别名的完整列表:

boolean: CBooleanValidator 的别名, 确保特性有一个 CBooleanValidator::trueValue 或CBooleanValidator::falseValue 值。

captcha: CCaptchaValidator 的别名,确保特性值等于 CAPTCHA 中显示的验证码。

compare: CCompareValidator 的别 名,确保特性等于另一个特性或常量。

email: CEmailValidator 的别名,确保特性是一个有效的Email地址。

default: CDefaultValueValidator 的别名,指定特性的默认值。

exist: CExistValidator 的别名,确保特性值可以在指定表的列中 可以找到。

file: CFileValidator 的别名,确保特性含有一个上传文件的名字。

filter: CFilterValidator 的别名,通 过一个过滤器改变此特性。

in: CRangeValidator 的别名,确保数据在一个预先指定的值的范围之内。

length: CStringValidator 的别名,确保数据的长度在一个指定的范围之内。

match: CRegularExpressionValidator 的别名,确保 数据可以匹配一个正则表达式。

numerical: CNumberValidator 的别名,确保数据是一个有效的数字。

required: CRequiredValidator 的别名,确保特性不为空。

type: CTypeValidator 的别名,确保特性是指定的数据类型。

unique: CUniqueValidator 的别名,确保数据在数据表的列中是唯一的。

url: CUrlValidator 的别名,确保数据是一个有效的 URL 。

下面我们列出了几个只用这些预定义验证器的示例:

// 用户名为必填项    
array('username', 'required'),    
// 用户名必须在 3 到 12 个字符之间    
array('username', 'length', 'min'=>3, 'max'=>12),    
// 在注册场景中,密码password必须和password2一致。    
array('password', 'compare', 'compareAttribute'=>'password2', 'on'=>'register'),    
// 在登录场景中,密码必须接受验证。    
array('password', 'authenticate', 'on'=>'login'),3. 安全的特性赋值

在一个类的实例被创建后,我 们通常需要用最终用户提交的数据填充它的特性。 这可以通过如下块赋值(massive assignment)方式轻松实现:

$model=new LoginForm;    
if(isset($_POST['LoginForm']))    
    $model->attributes=$_POST['LoginForm'];最后的表达式被称作 块赋值(massive assignment) ,它将 $_POST['LoginForm'] 中的每一项复制到相应的模型特性中。这相当于如下赋值方法:

foreach($_POST

['LoginForm'] as $name=>$value)    
{    
    if($name 是一个安全的特性)    
        $model->$name=$value;    
}检测特性的安全非常重要,例如,如果我们以为一个表的主键是安全的而暴露了它,那么攻击者可能就获得了一个修 改记录的主键的机会, 从而篡改未授权给他的内容。

检测特性安全的策略在版本 1.0 和 1.1 中是不同的,下面我们将 分别讲解:

1.1 中的安全特性

在版本 1.1 中,特性如果出现在相应场景的一个验证规则中,即被认为是安全的 。 例如:

array('username, password', 'required', 'on'=>'login, register'),    
array('email', 'required', 'on'=>'register'),如上所示, username 和 password 特性在 login 场景中是必 填项。而 username, password 和 email 特性在register 场景中是必填项。 于是,如果我们在 login 场景中执行块赋值,就 只有 username 和 password 会被块赋值。 因为只有它们出现在 login 的验证规则中。 另一方面,如果场景是 register , 这三个特性就都可以被块赋值。

// 在登录场景中    
$model=new User('login');    
if(isset($_POST['User']))    
    $model->attributes=$_POST['User'];    
        
// 在注册场景中    
$model=new User('register');    
if(isset($_POST['User']))    
    $model->attributes=$_POST['User'];那么为什么我们使用这样一种策略来检测特性是否安全呢? 背后的基 本原理就是:如果一个特性已经有了一个或多个可检测有效性的验证规则,那我们还担心什么呢?

请记住,验证规则是 用于检查用户输入的数据,而不是检查我们在代码中生成的数据(例如时间戳,自动产生的主键)。 因此,不要 为那些不接受 最终用户输入的特性添加验证规则。

有时候,我们想声明一个特性是安全的,即使我们没有为它指定任何规则。 例如, 一篇文章的内容可以接受用户的任何输入。我们可以使用特殊的 safe 规则实现此目的:

array('content', 'safe')

为了完成起见,还有一个用于声明一个属性为不安全的 unsafe 规则:

array ('permission', 'unsafe')

unsafe 规则并不常用,它是我们之前定义的安全特性的一个例外 。

1.0 中的安全特性

在版本1.0中,决定一个数据项是否是安全的,基于一个名为 safeAttributes 方法的返回值 和数据项被指定的场景. 默认的,这个方法返回所有公共成员变量作为 CFormModel 的安全特性,而它也返回了除了主键外, 表中 所有字段名作为 CActiveRecord的安全特性.我们可以根据场景重写这个方法来限制安全特性 .例如, 一个用户模型可以包含很 多特性,但是在 login 场景.里,我们只能使用 username 和 password 特性.我们可以按照如下来指定这一限制 :

public function safeAttributes()    
{    
    return array(    
        parent::safeAttributes(),    
        'login' => 'username, password',    
    );    
}safeAttributes 方法更准确的返回值应该是如下结构的 :

array(    
   // these attributes can be massively assigned in any scenario    
   // that is not explicitly specified below    
   'attr1, attr2, ...',    
     *    
   // these attributes can be massively assigned only in scenario 1    
   'scenario1' => 'attr2, attr3, ...',    
     *    
   // these attributes can be massively assigned only in scenario 2    
   'scenario2' => 'attr1, attr3, ...',    
)如果模型不是场景敏感的(比如,它只在一个场景中使用,或者所有场景共享了一套同样的安全特性),返 回值可以是如 下那样简单的字符串.

'attr1, attr2, ...'

而那些不安全的数据项,我们需要使用独立的赋值语句来分 配它们到相应的特性.如下所示:

$model->permission='admin';    
$model->id=1;4. 触发验证

一旦模型被用户提交的数据填充,我们就可以调用 CModel::validate() 出发 数据验证进程。此方法返回一个指示验证是否成功的值。 对 CActiveRecord 模型来说,验证也可以在我们调用其 CActiveRecord::save() 方法时自动触发。

我们可以使用 scenario 设置场景属性,这样,相应场景的验证规则就会被 应用。

验证是基于场景执行的。 scenario 属性指定了模型当前用于的场景和当前使用的验证规则集。 例如,在 login 场景中,我们只想验证用户模型中的 username 和 password 输入; 而在 register 场景中,我们需要验证更多的输入,例如 email, address, 等。 下面的例子演示了如何在 register 场景中执行验证:

// 在注册场景中创建一个  User 模型

。等价于:    
// $model=new User;    
// $model->scenario='register';    
$model=new User('register');    
        
// 将输入的值填充到模型    
$model->attributes=$_POST['User'];    
        
// 执行验证    
if($model->validate())   // if the inputs are valid    
    ...    
else
    ...规则关联的场景可以通过规则中的 on 选项指定。如果 on 选项未设置,则此规则会应用于所有场景。例如:

public function rules()    
{    
    return array(    
        array('username, password', 'required'),    
        array('password_repeat', 'required', 'on'=>'register'),    
        array('password', 'compare', 'on'=>'register'),    
    );    
}第一个规则将应用于所有场景,而第二个将只会应用于 register 场景。

5. 提取验证错误

验证完成 后,任何可能产生的错误将被存储在模型对象中。 我们可以通过调用 CModel::getErrors()和CModel::getError() 提取这些错 误信息。 这两个方法的不同点在于第一个方法将返回 所有 模型特性的错误信息,而第二个将只返回 第一个 错误信息。

6. 特性标签

当设计表单时,我们通常需要为每个表单域显示一个标签。 标签告诉用户他应该在此表单域中填写 什么样的信息。虽然我们可以在视图中硬编码一个标签, 但如果我们在相应的模型中指定(标签),则会更加灵活方便。

默认情况下 CModel 将简单的返回特性的名字作为其标签。这可以通过覆盖 attributeLabels() 方法自定义。 正如在 接下来的小节中我们将看到的,在模型中指定标签会使我们能够更快的创建出更强大的表单。

7. 创建动作Action方法

创建好LoginForm 表单Model后,我们就可以为它编写用户提交后的处理代码(对应到Controller中的某个Action方法) 。本例使用缺省的SiteController,对应的action为actionLogin.

public function actionLogin()    
{    
    $model=new LoginForm;    
    // collect user input data    
    if(isset($_POST['LoginForm']))    
    {    
        $model->attributes=$_POST['LoginForm'];    
        // validate user input and redirect to the previous page if valid    
        if($model->validate() && $model->login()){    
        
            $this->render('index');    
            return;    
        }    

    }    
    // display the login form    
    $this->render('login',array('model'=>$model));    
}如上所示,我们首先创建了一个 LoginForm 模型示例; 如果请求是一个 POST 请求(意味着这个登录表单被提交了 ),我们则使用提交的数据 $_POST['LoginForm'] 填充 $model ;然后我们验证此输入,如果验证成功,则显示index 页面。 如果验证失败,或者此动作被初次访问,我们则渲染 login 视图。
注意的我们修改了SiteController 的缺省 action为login.

/**   
 * @var string sets the default action to be 'login'   
 */
public $defaultAction='login';因此用户见到的第一个页面为login页面而非index页面,只有在用户输入正确的用 户名,本例使用固定的用户名和密码,参见UserIdentity类定义,实际应用可以读取数据库或是LDAP服务器。

 

/**   
 * UserIdentity represents the data needed to identity a user.   
 * It contains the authentication method that checks if the provided   
 * data can identity the user.   
 */
class UserIdentity extends CUserIdentity    
{    
    /**   
     * Authenticates a user.   
     * The example implementation makes sure if the username and password   
     * are both 'demo'.   
     * In practical applications, this should be changed to authenticate   
     * against some persistent user identity storage (e.g. database).   
     * @return boolean whether authentication succeeds.   
     */
    public function authenticate()    
    {    
        $users=array(    
            // username => password    
            'demo'=>'demo',    
            'admin'=>'admin',    
        );    
        if(!isset($users[$this->username]))    
            $this->errorCode=self::ERROR_USERNAME_INVALID;    
        else if($users[$this->username]!==$this->password)    
            $this->errorCode=self::ERROR_PASSWORD_INVALID;    
        else
            $this->errorCode=self::ERROR_NONE;    
        return !$this->errorCode;    
    }    
}让我们特别留意一下 login 动作中出现的下面的 PHP 语句:

$model->attributes=$_POST ['LoginForm'];

正如我们在 安全的特性赋值 中所讲的, 这行代码使用用户提交的数据填充模型。 attributes 属性由 CModel定义,它接受一个名值对数组并将其中的每个值赋给相应的模型特性。 因此如果 $_POST ['LoginForm'] 给了我们这样的一个数组,上面的那段代码也就等同于下面冗长的这段 (假设数组中存在所有所需的特 性):

$model->username=$_POST['LoginForm']['username'];
$model->password=$_POST ['LoginForm']['password'];
$model->rememberMe=$_POST['LoginForm'] ['rememberMe'];

8. 构建视图

编写 login 视图是很简单的,我们以一个 form 标记开始,它的 action 属性应该是前面讲述的 login 动作的URL。 然后我们需要为 LoginForm 类中声明的属性插入标签和表单域。最后, 我们插入 一个可由用户点击提交此表单的提交按钮。所有这些都可以用纯HTML代码完成。

Yii 提供了几个助手(helper)类简化 视图编写。例如, 要创建一个文本输入域,我们可以调用 CHtml::textField(); 要创建一个下拉列表,则调用 CHtml::dropDownList()。

信息: 你可能想知道使用助手的好处,如果它们所需的代码量和直接写纯HTML的代码量相当的 话。 答案就是助手可以提供比 HTML 代码更多的功能。例如, 如下代码将生成一个文本输入域,它可以在用户修改了其值时触 发表单提交动作。

CHtml::textField($name,$value,array('submit'=>''));

不然的话你就 需要写一大堆 Javascript 。

下面,我们使用 CHtml 创建一个登录表单。我们假设变量 $model 是 LoginForm 的实例 。

    
    

   

   


            
            
   

   

    
            
            
   

   

    
            
            
   

   

    
            
   

    

上述代码生成了一个更加动态的表单,例如, CHtml::activeLabel() 生成一个与 指定模型的特性相关的标签。 如果此特性有一个输入错误,此标签的CSS class 将变为 error,通过 CSS 样式改变了标签的外 观。 相似的,CHtml::activeTextField() 为指定模型的特性生成一个文本输入域,并会在错误发生时改变它的 CSS class。

如果我们使用由 yiic 脚本生提供的 CSS 样式文件,生成的表单就会像下面这样:

PHP开发框架Yii Framework教程(8) 使用FormModel

CSS 样式定义在css目录下,本例使用的为Yii缺省的样式。

从版本 1.1.1 开始,提供了一个新的小物件 CActiveForm 以简化表单创建。 这个小物件可同时提供客户端及服务器端无缝的、一致的验证。使用 CActiveForm, 上面的代 码可重写为:

    
beginWidget('CActiveForm'); ?>    
         
    errorSummary($model); ?>    
         
   
    
        label($model,'username'); ?>    
        textField($model,'username') ?>    
   
    
         
   
    
        label($model,'password'); ?>    
        passwordField($model,'password') ?>    
   
    
         
   
    
        checkBox($model,'rememberMe'); ?>    
        label($model,'rememberMe'); ?>    
   
    
         
   
    
            
   
    
         
endWidget(); ?>    
本例下载:http://www.imobilebbs.com/download/yii/LoginDemo.zip

从下篇开始将逐个介绍Yii框架支持的UI组件包括CActiveForm的用法。



推荐阅读
  • 本文介绍了如何通过安装 VirtualBox 和 Vagrant 来快速搭建和管理虚拟机环境。我们将详细探讨如何选择合适的 Box 镜像,以及如何高效地下载、添加和管理这些镜像。 ... [详细]
  • 本文介绍如何利用Python中的Epoll机制构建一个高效的Web服务器,该服务器能够处理多个并发连接,并向每个连接的客户端返回预定义的响应文本。通过使用Epoll,服务器可以实现高性能的I/O多路复用。 ... [详细]
  • 基于函数实现的进制转换工具
    本文介绍了一种利用函数实现不同进制数(二进制、八进制、十进制)之间转换的方法。包括了程序的运行效果展示、所使用的主要函数解析、以及如何验证用户输入的合法性。整个项目仅使用了两个全局变量来存储用户的选项和输入的数值。 ... [详细]
  • Java 动态代理详解与示例
    本文详细介绍了Java中的动态代理机制,包括如何定义接口、实现类和代理处理器,并通过具体示例演示了动态代理的创建和使用过程。 ... [详细]
  • 解决Android开发中的TextView难题
    探讨了在Android开发过程中遇到的关于TextView组件的常见问题,特别是如何实现多行文字的跑马灯效果,并提供了初步的解决方案和参考资料。 ... [详细]
  • Only2 Labs 是一家专注于视觉设计的工作室,如果您对当前的设计感到不满,或者急需寻找一个可靠的设计合作伙伴,甚至是您的团队项目需要专业指导,Only2 Labs 都将竭诚为您提供帮助。 ... [详细]
  • 本文探讨了Windows Presentation Foundation (WPF)如何通过扩展Microsoft Build Engine (MSBuild)来增强其构建能力,特别是在处理WPF特有的任务时。 ... [详细]
  • CGroups: 资源管理和控制
    CGroups(Control Groups)是Linux内核提供的一个功能,旨在限制、记录和隔离进程组使用的物理资源,如CPU、内存和I/O等。它通过精细的资源管理,支持现代容器技术如Docker的资源限制需求。 ... [详细]
  • 深入解析Java中的锁类型及其应用场景
    本文详细介绍了Java中常见的锁类型,包括乐观锁与悲观锁、独占锁与共享锁、互斥锁与读写锁、可重入锁、公平锁与非公平锁、分段锁、偏向锁、轻量级锁、重量级锁以及自旋锁。每种锁的特性、作用及适用场景均有所涉及。 ... [详细]
  • 本文探讨了HDU 4035的问题,涉及一个由n个房间组成的迷宫,这些房间通过n-1条隧道相互连接,形成一棵树结构。任务是从起点1号房间出发,计算到达出口所需经过的平均隧道数量,考虑了在每个房间中可能发生的三种情况及其相应概率。 ... [详细]
  • 一、数据更新操作DML语法中主要包括两个内容:查询与更新,更新主要包括:增加数据、修改数据、删除数据。其中这些操作是离不开查询的。1、增加数据语法:INSERTINTO表名称[(字 ... [详细]
  • HTML中用于创建表单的标签是什么
    本文将详细介绍HTML中用于创建表单的标签及其基本用法,包括表单的主要特性和常用的属性设置。如果您正在学习HTML或需要了解如何在网页中添加表单,这将是一个很好的起点。 ... [详细]
  • SQL注入实验:SqliLabs第38至45关解析
    本文深入探讨了SqliLabs项目中的第38至45关,重点讲解了堆叠注入(Stacked Queries)的应用技巧及防御策略。通过实际案例分析,帮助读者理解如何利用和防范此类SQL注入攻击。 ... [详细]
  • 解决fetch上传图片至微信公众号H5页面的问题
    在近期的一个项目需求中,需要在微信公众号内嵌入H5页面,并实现用户通过该页面上传图片的功能,包括拍摄新照片或从已有相册中选择。前端开发中采用了fetch API进行接口调用,但遇到了上传图片时数据无法正确传递的问题。 ... [详细]
  • 如何将Redis配置为后台服务
    本文介绍了在安装Redis后,如何通过修改配置文件使其以守护进程模式在后台运行,避免因控制台被占用而无法进行其他操作的问题。 ... [详细]
author-avatar
大爱仅有的财产丶_468
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有