Yii快速入门经典教程
本文讲述了Yii快速入门教程。分享给大家供大家参考,具体如下:
Ⅰ、基本概念
一、入口文件
入口文件内容:一般格式如下:
<?php $yii=dirname(__FILE__).'/../../framework/yii.php';//Yii框架位置 $config=dirname(__FILE__).'/protected/config/main.php';//当前应用程序的主配置文件位置 //部署正式环境时,去掉下面这行 //defined('YII_DEBUG')ordefine('YII_DEBUG',true);//是否运行在调试模式下 require_once($yii);//包含Yii框架 Yii::createWebApplication($config)->run();//根据主配置文件建立应用实例,并运行。你可以在当前应用的任何位置通过Yii::app()来访问这个实例。
二、主配置文件
保存位置:你的应用/protected/config/main.php
文件内容:一般格式如下:
<?php returnarray( 'basePath'=>dirname(__FILE__).DIRECTORY_SEPARATOR.'..',//当前应用根目录的绝对物理路径 'name'=>'YiiBlogDemo',//当前应用的名称 //预载入log(记录)应用组件,这表示该应用组件无论它们是否被访问都要被创建。该应用的参数配置在下面以“components”为关键字的数组中设置。 'preload'=>array('log'),//log为组件ID //自动载入的模型和组件类 'import'=>array( 'application.models.*',//载入“application/models/”文件夹下的所有模型类 'application.components.*',//载入“application/components/”文件夹下的所有应用组件类 ), 'defaultController'=>'post',//设置默认控制器类 //当前应用的组件配置。更多可供配置的组件详见下面的“核心应用组件” 'components'=>array( 'user'=>array(//user(用户)组件配置,“user”为组件ID //可以使用基于cookie的认证 'allowAutoLogin'=>true,//允许自动登录 ), 'cache'=>array(//缓存组件 'class'=>'CMemCache',//缓存组件类 'servers'=>array(//MemCache缓存服务器配置 array('host'=>'server1','port'=>11211,'weight'=>60),//缓存服务器1 array('host'=>'server2','port'=>11211,'weight'=>40),//缓存服务器2 ), ), 'db'=>array(//db(数据库)组件配置,“db”为组件ID 'connectionString'=>'sqlite:protected/data/blog.db',//连接数据库的DSN字符串 'tablePrefix'=>'tbl_',//数据表前缀 ), //如果要使用一个MySQL数据库,请取消下面的注释 'errorHandler'=>array( //使用SiteController控制器类中的actionError方法显示错误 'errorAction'=>'site/error',//遇到错误时,运行的操作。控制器名和方法名均小写,并用斜线“/”隔开 ), //URL路由管理器 'urlManager'=>array( 'urlFormat'=>'path',//URL格式。共支持两种格式:'path'格式(如:/path/to/EntryScript.php/name1/value1/name2/value2...)和'get'格式(如:/path/to/EntryScript.php?name1=value1&name2=value2...)。当使用'path'格式时,需要设置如下的规则: 'rules'=>array(//URL规则。语法:<参数名:正则表达式> 'post/<id:\d+>/<title:.*?>'=>'post/view',//将post/12/helloword指向post/view?id=12&title=helloword 'posts/<tag:.*?>'=>'post/index',//将posts/hahahaha指向post/index?tag=hahahaha '<controller:\w+>/<action:\w+>'=>'<controller>/<action>', ), ), 'log'=>array(//记录 'class'=>'CLogRouter',//处理记录信息的类 'routes'=>array( array( 'class'=>'CFileLogRoute',//处理错误信息的类 'levels'=>'error,warning',//错误等级 ), //如要将错误记录消息在网页上显示,取消下面的注释即可 ), ), ),//应用组件配置结束 //使用Yii::app()->params['参数名']可以访问应用层的参数 'params'=>require(dirname(__FILE__).'/params.php'), );
核心应用组件:
Yii预定义了一系列核心应用组件,提供常见Web应用中所用的功能。例如,request组件用于解析用户请求并提供例如URL,cookie等信息。通过配置这些核心组件的属性,我们可以几乎任意的修改Yii的默认行为。
下面我们列出了由CWebApplication预定义的核心组件。
assetManager:CAssetManager-管理私有资源文件的发布。
authManager:CAuthManager-管理基于角色的访问控制(RBAC).
cache:CCache-提供数据缓存功能。注意,你必须指定实际的类(例如CMemCache,CDbCache)。否则,当你访问此组件时将返回NULL。
clientScript:CClientScript-管理客户端脚本(javascripts和CSS).
coreMessages:CPhpMessageSource-提供Yii框架用到的核心信息的翻译。
db:CDbConnection-提供数据库连接。注意,使用此组件你必须配置其connectionString属性。
errorHandler:CErrorHandler-处理未捕获的PHP错误和异常。
format:CFormatter-格式化数值显示。此功能从版本1.1.0起开始提供。
messages:CPhpMessageSource-提供Yii应用中使用的信息翻译。
request:CHttpRequest-提供关于用户请求的信息。
securityManager:CSecurityManager-提供安全相关的服务,例如散列,加密。
session:CHttpSession-提供session相关的功能。
statePersister:CStatePersister-提供全局状态持久方法。
urlManager:CUrlManager-提供URL解析和创建相关功能
user:CWebUser-提供当前用户的识别信息。
themeManager:CThemeManager-管理主题。
要访问一个应用组件,使用Yii::app()->组件的ID
三、控制器(Controller)
控制器是CController类的子类的实例。它在当用户请求时由应用创建。当一个控制器运行时,它执行所请求的动作(控制器类方法),动作通常会引入所必要的模型并渲染相应的视图。动作,就是一个名字以action开头的控制器类方法(action+大写首字母的动作名)。
控制器类文件保存位置protected/controllers/
控制器和动作以ID识别。
控制器ID是一种'父目录/子目录/控制器名'的格式,对应相应的控制器类文件protected/controllers/父目录/子目录/大写首字母的控制器名Controller.php;
动作ID是除去action前缀的动作方法名。
1、路由
用户以路由的形式请求特定的控制器和动作。路由是由控制器ID和动作ID连接起来的,两者以斜线分割。
例如,路由post/edit代表PostController及其edit动作。默认情况下,URLhttp://hostname/index.php?r=post/edit即请求此控制器和动作。
注意:默认情况下,路由是大小写敏感的。可以通过设置应用配置中的CUrlManager::caseSensitive为false使路由对大小写不敏感。当在大小写不敏感模式中时,要确保你遵循了相应的规则约定,即:包含控制器类文件的目录名小写,且控制器映射和动作映射中使用的键为小写。
路由的格式:控制器ID/动作ID或模块ID/控制器ID/动作ID(如果是嵌套模块,模块ID就是父模块ID/子模块ID)
2、控制器实例化
应用将使用如下规则确定控制器的类以及类文件的位置:
①、如果指定了CWebApplication::catchAllRequest,控制器将基于此属性创建,而用户指定的控制器ID将被忽略。这通常用于将应用设置为维护状态并显示一个静态提示页面。
②、如果在CWebApplication::controllerMap中找到了ID,相应的控制器配置将被用于创建控制器实例。
③、如果ID为'path/to/xyz'的格式,控制器类的名字将判断为XyzController,相应的类文件则为protected/controllers/path/to/XyzController.php。如果类文件不存在,将触发一个404CHttpException异常。
在使用了模块的情况下,应用将检查此ID是否代表一个模块中的控制器。如果是的话,模块实例将被首先创建,然后创建模块中的控制器实例。
3、动作(action)
动作就是被定义为一个以action单词作为前缀命名的方法。而更高级的方式是定义一个动作类并让控制器在收到请求时将其实例化。这使得动作可以被复用,提高了可复用度。
1、定义一个动作类,基本格式如下:
classUpdateActionextendsCAction { publicfunctionrun() { //placetheactionlogichere } }
2、使用动作类:为了让控制器注意到这个动作,我们要用如下方式覆盖控制器类的actions()方法:
classPostControllerextendsCController { publicfunctionactions() { returnarray( 'edit'=>'application.controllers.post.UpdateAction',//使用“应用程序文件夹/controllers/post/UpdateAction.php”文件中的类来处理edit动作 ); } }
如上所示,我们使用了路径别名“application.controllers.post.UpdateAction”指定动作类文件为“protected/controllers/post/UpdateAction.php”。
通过编写基于类的动作,我们可以将应用组织为模块的风格。例如,如下目录结构可用于组织控制器相关代码:
protected/
controllers/
PostController.php
UserController.php
post/
CreateAction.php
ReadAction.php
UpdateAction.php
user/
CreateAction.php
ListAction.php
ProfileAction.php
UpdateAction.php
4、过滤器(filter)
过滤器是一段代码,可被配置在控制器动作执行之前或之后执行。
一个动作可以有多个过滤器。如有多个过滤器,则按照它们出现在过滤器列表中的顺序依次执行。过滤器可以阻止动作及后面其他过滤器的执行。
过滤器可以定义为一个控制器类的方法。过滤器方法名必须以filter开头。例如,现有的filterAccessControl方法定义了一个名为accessControl的过滤器。过滤器方法必须为如下结构:
publicfunctionfilterAccessControl($filterChain) { //调用$filterChain->run()以继续后续过滤器与动作的执行。 }
$filterChain(过滤器链)是一个CFilterChain的实例,代表与所请求动作相关的过滤器列表。在过滤器方法中,我们可以调用$filterChain->run()以继续执行后续过滤器和动作。
如动作一样,过滤器也可以是一个对象,它是CFilter或其子类的实例。如下代码定义了一个新的过滤器类:
classPerformanceFilterextendsCFilter { protectedfunctionpreFilter($filterChain) { //动作被执行之前应用的逻辑 returntrue;//如果动作不应被执行,此处返回false } protectedfunctionpostFilter($filterChain) { //动作执行之后应用的逻辑 } }
要对动作应用过滤器,我们需要覆盖CController::filters()方法。此方法应返回一个过滤器配置数组。例如:
classPostControllerextendsCController { ...... publicfunctionfilters() { returnarray( 'postOnly+edit,create',//将postOnly过滤器应用于edit和create动作(这是基于方法的过滤器) array(//使用了数组来配置过滤器 'application.filters.PerformanceFilter-edit,create',//将application.filters.PerformanceFilter过滤器应用于除了edit和create之外的所有动作(这是基于对象的过滤器) 'unit'=>'second',//初始化过滤器对象中的unit属性值为second ), ); } }
上述代码指定了两个过滤器:postOnly和PerformanceFilter。postOnly过滤器是基于方法的(相应的过滤器方法已在CController中定义);而performanceFilter过滤器是基于对象的。路径别名application.filters.PerformanceFilter指定过滤器类文件是protected/filters/PerformanceFilter。我们使用一个数组配置PerformanceFilter,这样它就可被用于初始化过滤器对象的属性值。此处PerformanceFilter的unit属性值将被初始为second。
使用加减号,我们可指定哪些动作应该或不应该应用过滤器。上述代码中,postOnly应只被应用于edit和create动作,而PerformanceFilter应被应用于除了edit和create之外的动作。如果过滤器配置中没有使用加减号,则此过滤器将被应用于所有动作。
五、模型(Model)
模型是CModel或其子类的实例。模型用于保持数据以及与其相关的业务逻辑。
模型是单独的数据对象。它可以是数据表中的一行,或者一个用户输入的表单。
数据对象的每个字段对应模型中的一个属性。每个属性有一个标签(label),并且可以通过一系列规则进行验证。
Yii实现了两种类型的模型:表单模型和ActiveRecord。二者均继承于相同的基类CModel。
表单模型是CFormModel的实例。表单模型用于保持从用户的输入获取的数据。这些数据经常被获取,使用,然后丢弃。例如,在一个登录页面中,我们可以使用表单模型用于表示由最终用户提供的用户名和密码信息。
ActiveRecord(AR)是一种用于通过面向对象的风格抽象化数据库访问的设计模式。每个AR对象是一个CActiveRecord或其子类的实例。代表数据表中的一行。行中的字段对应AR对象中的属性。
六、视图
视图是一个包含了主要的用户交互元素的PHP脚本.
视图有一个名字,当渲染(render)时,名字会被用于识别视图脚本文件。视图的名称与其视图脚本名称是一样的。例如:视图edit的名称出自一个名为edit.php的脚本文件。要渲染时,需通过传递视图的名称调用CController::render()。这个方法将在“protected/views/控制器ID”目录下寻找对应的视图文件。
在视图脚本内部,我们可以通过$this来访问控制器实例。我们可以在视图里以“$this->属性名”的方式获取控制器的任何属性。
我们也可以用以下推送的方式传递数据到视图里:
$this->render('edit',array( 'var1'=>$value1, 'var2'=>$value2, ));
在以上的方式中,render()方法将提取数组的第二个参数到变量里。其产生的结果是,在视图脚本里,我们可以直接访问变量$var1和$var2。
1、布局
布局是一种用来修饰视图的特殊的视图文件。它通常包含了用户界面中通用的一部分视图。例如:布局可以包含header和footer的部分,然后把内容嵌入其间。
......headerhere...... <?phpecho$content;?> ......footerhere......
其中的$content则储存了内容视图的渲染结果。
当使用render()时,布局被隐式应用。视图脚本protected/views/layouts/main.php是默认的布局文件。这可以通过改变CWebApplication::layout进行自定义。要渲染一个不带布局的视图,则需调用renderPartial()。
2、小物件
小物件是CWidget或其子类的实例。它是一个主要用于表现数据的组件。小物件通常内嵌于一个视图来产生一些复杂而独立的用户界面。例如,一个日历小物件可用于渲染一个复杂的日历界面。小物件使用户界面更加可复用。
我们可以按如下视图脚本来使用一个小物件:
<?php$this->beginWidget('小物件类的路径别名'[,'包含属性初始化值的数组']);?> ...可能会由小物件获取的内容主体... <?php$this->endWidget();?>
或者
<?php$this->widget('小物件类的路径别名'[,'包含属性初始化值的数组']);?>
后者用于不需要任何body内容的组件。
小物件可通过配置来定制它的表现。这是通过调用CBaseController::beginWidget或CBaseController::widget设置其初始化属性值来完成的。
我们通过传递一个携带这些属性初始化值的数组来实现,该数组的键是属性的名称,而数组的值则是小物件属性所对应的值。如下所示:
<?php $this->widget('CMaskedTextField',array( 'mask'=>'99/99/9999' )); ?>
继承CWidget并覆盖其init()和run()方法,可以定义一个新的小物件:
classMyWidgetextendsCWidget { publicfunctioninit() { //此方法会被CController::beginWidget()调用 } publicfunctionrun() { //此方法会被CController::endWidget()调用 } }
小物件可以像一个控制器一样拥有它自己的视图。
默认情况下,小物件的视图文件位于包含了小物件类文件目录的views子目录之下(protected/components/views)。这些视图可以通过调用CWidget::render()渲染,这一点和控制器很相似。唯一不同的是,小物件的视图没有布局文件支持。另外,小物件视图中的$this指向小物件实例而不是控制器实例。
3、系统视图
系统视图的渲染通常用于展示Yii的错误和日志信息。
系统视图的命名遵从了一些规则。比如像“errorXXX”这样的名称就是用于渲染展示错误号XXX的CHttpException的视图。例如,如果CHttpException抛出一个404错误,那么error404就会被显示。
在framework/views下,Yii提供了一系列默认的系统视图.他们可以通过在protected/views/system下创建同名视图文件进行自定义。
七、组件
Yii应用建立于组件之上。组件是CComponent或其子类的实例。使用组件主要涉及访问它的属性以及触发或处理它的时间。基类CComponent指定了如何定义属性和事件。
1、组件属性
组件的属性就像对象的公共成员变量。它是可读写的。
要定义一个组件属性,我们只需在组件类中定义一个公共成员变量即可。
更灵活的方式是定义其getter和setter方法,例如:
publicfunctiongetTextWidth()//获取textWidth属性 { return$this->_textWidth; } publicfunctionsetTextWidth($value)//设置TextWidth属性 { $this->_textWidth=$value; }
上述代码定义了一个可写的属性名为textWidth(名字是大小写不敏感的)。当读取属性时,getTextWidth()就会被调用,其返回值则成为属性值;相似的,当写入属性时,setTextWidth()被调用。如果setter方法没有定义,则属性将是只读的,如果对其写入则会抛出一个异常。使用getter和setter方法定义一个属性有一个好处:即当读取或写入属性时,可以执行额外的逻辑(例如,执行验证,触发事件)。
注意:通过getter/setter定义的属性和类成员变量之间有一个细微的差异:属性的名字是大小写不敏感的,而类成员变量是大小写敏感的。
2、组件事件
组件事件是一些特殊的属性,它们使用一些称作事件句柄(eventhandlers)的方法作为其值。分配一个方法到一个事件将会引起方法在事件被唤起处自动被调用。因此,一个组件的行为可能会被一种在部件开发过程中不可预见的方式修改。
组件事件以on开头的命名方式定义。和属性通过getter/setter方法来定义的命名方式一样,事件的名称是大小写不敏感的。以下代码定义了一个onClicked事件:
publicfunctiononClicked($event) { $this->raiseEvent('onClicked',$event); }
这里作为事件参数的$event是CEvent或其子类的实例。
我们可以分配一个方法到此事件,如下所示:
$component->onClicked=$callback;
这里的$callback指向了一个有效的PHP回调。它可以是一个全局函数也可以是类中的一个方法。如果是后者,它必须以一个数组的方式提供:array($object,'methodName')。
事件句柄的结构如下:
function方法名($event) { ...... }
这里的$event即描述事件的参数(它来源于raiseEvent()调用)。$event参数是CEvent或其子类的实例。至少,它包含了关于谁触发了此事件的信息。
事件句柄也可以是一个PHP5.3以后支持的匿名函数。例如:
$component->onClicked=function($event){ ...... }
如果我们现在调用onClicked(),onClicked事件将被触发(在onClicked()中),附属的事件句柄将被自动调用。
一个事件可以绑定多个句柄。当事件触发时,这些句柄将被按照它们绑定到事件时的顺序依次执行。如果句柄决定组织后续句柄被执行,它会设置$event->handled为true。
3、组件行为
组件已添加了对mixin的支持,并可以绑定一个或多个行为。行为是一个对象,其方法可以被它绑定的部件通过收集功能的方式来实现继承(inherited),而不是专有化继承(即普通的类继承)。一个部件可以以'多重继承'的方式实现多个行为的绑定。
行为类必须实现IBehavior接口。大多数行为可以继承自CBehavior。如果一个行为需要绑定到一个模型,它也可以从专为模型实现绑定特性的CModelBehavior或CActiveRecordBehavior继承。
要使用一个行为,它必须首先通过调用此行为的attach()方法绑定到一个组件。然后我们就可以通过组件调用此行为方法:
//$name在组件中实现了对行为的唯一识别 $component->attachBehavior($name,$behavior); //test()是行为中的方法。 $component->test();
已绑定的行为可以像一个组件中的普通属性一样访问。例如,如果一个名为tree的行为绑定到了一个组件,我们就可以通过如下代码获得指向此行为的引用。
$behavior=$component->tree; //等于下行代码: //$behavior=$component->asa('tree');
行为是可以被临时禁止的,此时它的方法就会在组件中失效。例如:
$component->disableBehavior($name); //下面的代码将抛出一个异常 $component->test(); $component->enableBehavior($name); //现在就可以使用了 $component->test();
两个同名行为绑定到同一个组件下是有可能的。在这种情况下,先绑定的行为则拥有优先权。
当和events,一起使用时,行为会更加强大。当行为被绑定到组件时,行为里的一些方法就可以绑定到组件的一些事件上了。这样一来,行为就有机观察或者改变组件的常规执行流程。
一个行为的属性也可以通过绑定到的组件来访问。这些属性包含公共成员变量以及通过getters和/或setters方式设置的属性。例如,若一个行为有一个xyz的属性,此行为被绑定到组件$a,然后我们可以使用表达式$a->xyz访问此行为的属性。
八、模块
模块是一个独立的软件单元,它包含模型,视图,控制器和其他支持的组件。在许多方面上,模块看起来像一个应用。主要的区别就是模块不能单独部署,它必须存在于一个应用里。用户可以像他们访问普通应用的控制器那样访问模块中的控制器。
模块在一些场景里很有用。对大型应用来说,我们可能需要把它划分为几个模块,每个模块可以单独维护和部署。一些通用的功能,例如用户管理,评论管理,可以以模块的形式开发,这样他们就可以容易地在以后的项目中被复用。
1、创建模块
模块组织在一个目录中,目录名即为模块的唯一ID。模块目录的结构跟应用基础目录很相似。下面列出了一个fourm的模块的典型的目录结构:
forum/模块文件夹
ForumModule.php模块类文件
components/包含可复用的用户组件
views/包含小物件的视图文件
controllers/包含控制器类文件
DefaultController.php默认的控制器类文件
extensions/包含第三方扩展
models/包含模型类文件
views/包含控制器视图和布局文件
layouts/包含布局文件
default/包含DefaultController的视图文件
index.php首页视图文件
模块必须有一个继承自CWebModule的模块类。类的名字通过表达式ucfirst($id).'Module'确定,其中的$id代表模块的ID(或者说模块的目录名字)。模块类是存储模块代码间可共享信息的中心位置。例如,我们可以使用CWebModule::params存储模块参数,使用CWebModule::components分享模块级的应用组件。
2、使用模块
要使用模块,首先将模块目录放在应用基础目录的modules文件夹中。然后在应用的modules属性中声明模块ID。例如,为了使用上面的forum模块,我们可以使用如下应用配置:
returnarray( ...... 'modules'=>array('forum',...), ...... );
模块也可以在配置时带有初始属性值。做法和配置应用组件很类似。例如,forum模块可以在其模块类中有一个名为postPerPage的属性,它可以在应用配置中配置如下:
returnarray( ...... 'modules'=>array( 'forum'=>array( 'postPerPage'=>20, ), ), ...... );
模块的实例可通过当前活动控制器的module属性访问。在模块实例中,我们可以访问在模块级中共享的信息。例如,为访问上面的postPerPage信息,我们可使用如下表达式:
$postPerPage=Yii::app()->controller->module->postPerPage; //如如$this引用的是控制器实例,则可以使用下行语句 //$postPerPage=$this->module->postPerPage;
模块中的控制器动作可以通过路由“模块ID/控制器ID/动作ID”或“模块ID/存放控制器类文件的子目录名/控制器ID/动作ID”访问。例如,假设上面的forum模块有一个名为PostController的控制器,我们就可以通过路由forum/post/create访问此控制器中的create动作。此路由对应的URL即http://www.example.com/index.php?r=forum/post/create。
3、嵌套的模块
模块可以无限级嵌套。这就是说,一个模块可以包含另一个模块,而这另一个模块又可以包含其他模块。我们称前者为父模块,后者为子模块。子模块必须定义在其父模块的modules属性中,就像我们前面在应用配置中定义模块一样。
要访问子模块中的控制器动作,我们应使用路由父模块ID/子模块ID/控制器ID/动作ID。
九、路径别名
Yii中广泛的使用了路径别名。路径别名关联于一个目录或文件的路径。它以点号语法指定,类似于广泛使用的名字空间(namespace)格式:
RootAlias.path.to.target
其中的RootAlias是某个现存目录的别名,通过调用YiiBase::setPathOfAlias(),我们可以定义新的路径别名。为方便起见,Yii预定义了以下几个根别名:
system:表示Yii框架目录;
zii:表示Zii库目录;
application:表示应用的基础目录;
webroot:表示入口脚本文件所在的目录。
ext:表示包含了所有第三方扩展的目录。
额外的,如果应用使用了模块,(Yii)也为每个模块ID定义了根别名,指向相应模块的跟目录。
通过使用YiiBase::getPathOfAlias(),别名可以被翻译为其相应的路径。
使用别名可以很方便的导入类的定义。例如,如果我们想包含CController类的定义,我们可以调用如下代码
Yii::import('system.web.CController');
import方法跟include和require不同,它更加高效。导入(import)的类定义并不会真正被包含进来,直到它第一次被引用。多次导入同样的名字空间也会比include_once和require_once快得多。
我们还可以使用如下语法导入整个目录,这样此目录下的类文件就会在需要时被自动包含。
Yii::import('system.web.*');
除import外,别名还在其他许多地方指向类。例如,路径别名可以传递给Yii::createComponent()以创建相应类的实例。即使类文件在之前从未被包含。
不要将路径别名和名字空间混淆了,名字空间是指对一些类名的一个逻辑组合,这样它们就可以相互区分开,即使有相同的名字。而路径别名是用于指向一个类文件或目录。路径别名与名字空间并不冲突。
十、开发规范
下面我们讲解Yii编程中推荐的开发规范。为简单起见,我们假设WebRoot是Yii应用安装的目录。
1、URL
默认情况下,Yii识别如下格式的URL:
http://hostname/index.php?r=ControllerID/ActionID
r变量意为路由(route),它可以被Yii解析为控制器和动作。如果ActionID被省略,控制器将使用默认的动作(在CController::defaultAction中定义);如果ControllerID也被省略(或者r变量不存在),应用将使用默认的控制器(在CWebApplication::defaultController中定义)。
通过CUrlManager的帮助,可以创建更加可识别,更加SEO友好的URL,例如http://hostname/ControllerID/ActionID.html。
2、代码
Yii推荐命名变量、函数和类时使用驼峰风格,即每个单词的首字母大写并连在一起,中间无空格。变量名和函数名应该使它们的第一个单词全部小写,以使其区别于类名。对私有类成员变量来说,我们推荐以下划线作为其名字前缀(例如:$_actionList)。
一个针对控制器类名的特殊规则是它们必须以单词Controller结尾。那么控制器ID就是类名的首字母小写并去掉单词Controller。例如,PageController类的ID就是page。这个规则使应用更加安全。它还使控制器相关的URL更加简单(例如/index.php?r=page/index而不是/index.php?r=PageController/index)。
3、配置
配置是一个键值对数组。每个键代表了所配置的对象中的属性名,每个值则为相应属性的初始值。
类中任何可写的属性都可以被配置。如果没有配置,属性将使用它们的默认值。当配置一个属性时,最好阅读相应文档以保证初始值正确。
4、文件
命名和使用文件的规范取决于它们的类型。
类文件应以它们包含的公有类命名。例如,CController类位于CController.php文件中。公有类是可以被任何其他类使用的类。每个类文件应包含最多一个公有类。私有类(只能被一个公有类使用的类)可以放在使用此类的公有类所在的文件中。
视图文件应以视图的名字命名。例如,index视图位于index.php文件中。视图文件是一个PHP脚本文件,它包含了用于呈现内容的HTML和PHP代码。
配置文件可以任意命名。配置文件是一个PHP脚本,它的主要目的是返回一个体现配置的关联数组。
5、目录
Yii假定了一系列默认的目录用于不同的场合。如果需要,每个目录都可以自定义。
WebRoot/protected:这是应用基础目录,是放置所有安全敏感的PHP脚本和数据文件的地方。Yii有一个默认的application别名指向此目录。此目录及目录中的文件应该保护起来防止Web用户访问。它可以通过CWebApplication::basePath自定义。
WebRoot/protected/runtime:此目录放置应用在运行时产生的私有临时文件。此目录必须对Web服务器进程可写。它可以通过CApplication::runtimePath自定义。
WebRoot/protected/extensions:此目录放置所有第三方扩展。它可以通过CApplication::extensionPath自定义。
WebRoot/protected/modules:此目录放置所有的应用模块,每个模块使用一个子目录。
WebRoot/protected/controllers:此目录放置所有控制器类文件。它可以通过CWebApplication::controllerPath自定义。
WebRoot/protected/views:此目录放置所有试图文件,包含控制器视图,布局视图和系统视图。它可以通过CWebApplication::viewPath自定义。
WebRoot/protected/views/ControllerID:此目录放置单个控制器类中使用的视图文件。此处的ControllerID是指控制器的ID。它可以通过CController::viewPath自定义。
WebRoot/protected/views/layouts:此目录放置所有布局视图文件。它可以通过CWebApplication::layoutPath自定义。
WebRoot/protected/views/system:此目录放置所有系统视图文件。系统视图文件是用于显示异常和错误的模板。它可以通过CWebApplication::systemViewPath自定义。
WebRoot/assets:此目录放置公共资源文件。资源文件是可以被发布的,可由Web用户访问的私有文件。此目录必须对Web服务器进程可写。它可以通过CAssetManager::basePath自定义
WebRoot/themes:此目录放置应用使用的不同的主题。每个子目录即一个主题,主题的名字即目录的名字。它可以通过CThemeManager::basePath自定义。
6、数据库
多数Web应用是由数据库驱动的。我们推荐在对表和列命名时使用如下命名规范。注意,这些规范并不是Yii所必须的。
㈠数据库表名和列名都使用小写命名。
㈡名字中的单词应使用下划线分割(例如product_order)。
㈢对于表名,你既可以使用单数也可以使用复数。但不要同时使用两者。为简单起见,我们推荐使用单数名字。
㈣表名可以使用一个通用前缀,例如tbl_。这样当应用所使用的表和另一个应用说使用的表共存于同一个数据库中时就特别有用。这两个应用的表可以通过使用不同的表前缀很容易地区别开。
Ⅱ、使用表单
在Yii中处理表单时,通常需要以下步骤:
1.创建用于表现所要收集数据字段的模型类。
2.创建一个控制器动作,响应表单提交。
3.在视图脚本中创建与控制器动作相关的表单。
一、创建模型
在编写表单所需的HTML代码之前,我们应该先确定来自最终用户输入的数据的类型,以及这些数据应符合什么样的规则。模型类可用于记录这些信息。正如模型章节所定义的,模型是保存用户输入和验证这些输入的中心位置。
取决于使用用户所输入数据的方式,我们可以创建两种类型的模型。如果用户输入被收集、使用然后丢弃,我们应该创建一个表单模型;如果用户的输入被收集后要保存到数据库,我们应使用一个ActiveRecord。两种类型的模型共享同样的基类CModel,它定义了表单所需的通用接口。
1、定义模型类
例如创建为一个表单模型:
classLoginFormextendsCFormModel { public$username; public$password; public$rememberMe=false; }
LoginForm中定义了三个属性:$username,$password和$rememberMe。他们用于保存用户输入的用户名和密码,还有用户是否想记住他的登录的选项。由于$rememberMe有一个默认的值false,相应的选项在初始化显示在登录表单中时将是未勾选状态。
我们将这些成员变量称为特性(attributes)而不是属性(properties),以区别于普通的属性(properties)。特性(attribute)是一个主要用于存储来自用户输入或数据库数据的属性(propertiy)。
2、声明验证规则
一旦用户提交了他的输入,模型被填充,我们就需要在使用前确保用户的输入是有效的。这是通过将用户的输入和一系列规则执行验证实现的。我们在rules()方法中指定这些验证规则,此方法应返回一个规则配置数组。
classLoginFormextendsCFormModel { public$username; public$password; public$rememberMe=false; private$_identity; publicfunctionrules() { returnarray( array('username,password','required'),//username和password为必填项 array('rememberMe','boolean'),//rememberMe应该是一个布尔值 array('password','authenticate'),//password应被验证(authenticated) ); } publicfunctionauthenticate($attribute,$params) { $this->_identity=newUserIdentity($this->username,$this->password); if(!$this->_identity->authenticate()) $this->addError('password','错误的用户名或密码。'); } }
rules()返回的每个规则必须是以下格式:
array('AttributeList','Validator','on'=>'ScenarioList',...附加选项)
其中:
AttributeList(特性列表)是需要通过此规则验证的特性列表字符串,每个特性名字由逗号分隔;
Validator(验证器)指定要执行验证的种类;
on参数是可选的,它指定此规则应被应用到的场景列表;
附加选项是一个名值对数组,用于初始化相应验证器的属性值。
希望本文所述对大家基于Yii框架的PHP程序设计有所帮助。