澳门新葡萄京娱乐场Yii使用技巧大汇总

在 Yii 自动生成的代码里,我们总能在 admin 的界面看到 CGridView
的身影。这是一个很好用的展示数据的表格控件,用的好可以明显地加快开发进度。下面就让我们来探索一下
CGridView 的基本使用吧:

本文汇总了Yii使用技巧。分享给大家供大家参考,具体如下:

简单起见,我们的代码就用 Yii demo 中的 blog
例子来做修改。首先,这是修改后的部分 Mysql 语句:

一、从yii权威指中学到的

drop table if exists `tbl_user`; CREATE TABLE tbl_user ( `user_id` INTEGER NOT NULL AUTO_INCREMENT comment '主键', `username` VARCHAR(128) NOT NULL comment '用户名', `nickname` VARCHAR(128) NOT NULL comment '昵称', `password` VARCHAR(128) NOT NULL comment '密码', `email` VARCHAR(128) NOT NULL comment '邮箱', `is_delete` tinyint not null default 0 comment '删除标志', unique key(`username`), primary key (`user_id`) ) ENGINE=InnoDB DEFAULT CHARSET=utf8 comment='用户表'; drop table if exists `tbl_post`; CREATE TABLE tbl_post ( `post_id` INTEGER NOT NULL AUTO_INCREMENT comment '主键', `title` VARCHAR(128) NOT NULL comment '标题', `content` TEXT NOT NULL comment '文章内容', `tags` TEXT comment '标签', `status` INTEGER NOT NULL comment '状态,0 = 草稿,1 = 审核通过,-1 = 审核不通过,2 = 发布', `create_time` INTEGER comment '创建时间', `update_time` INTEGER comment '更新时间', `author_id` INTEGER NOT NULL comment '作者', `is_delete` tinyint not null default 0 comment '删除标志', CONSTRAINT `post_ibfk_1` FOREIGN KEY (author_id) REFERENCES tbl_user (`user_id`) ON DELETE CASCADE ON UPDATE RESTRICT, primary key (`post_id`) ) ENGINE=InnoDB DEFAULT CHARSET=utf8 comment='日志表'; 

db组件 ‘schemaCachingDuration’=>3600, 为什么不起做用?

两个表一个存储作者信息一个存储日志,其中日志有一个外键关联到
user。两个表里面的 is_delete 字段是标志该条记录是否被删除,0
为未删除,1 为已删除。让我们看一下用 gii 生成的 Post 类的 relation
方法:

需要开缓存

/** * @return array relational rules. */ public function relations() { // NOTE: you may need to adjust the relation name and the related // class name for the relations automatically generated below. return array( 'comments' = array(self::HAS_MANY, 'Comment', 'post_id'), 'author' = array(self::BELONGS_TO, 'User', 'author_id'), ); } 

如何在页面下边显示sql的查询时间

其中的 author 外键作为 BELONGS_TO 关系存在,符合我们的预期。
说了这么多,看看自动生成的 Post 中 admin.php 里 CGridView 的代码吧:

在log组件的routes中加入

php $this-widget('zii.widgets.grid.CGridView', array( 'id'='post-grid', 'dataProvider'=$model-search(), 'filter'=$model, 'columns'=array( 'post_id', 'title', 'content', 'tags', 'status', 'create_time', 'update_time', 'author_id', 'is_delete', array( 'class'='CButtonColumn', ), ), ));  
array(
'class'=>'CProfileLogRoute',
'levels'=>'error, warning',
)

看!虽然我们什么都没写,但这就是这个控件的最基础使用了。dataProvider
是由 model 里面的 search
函数提供的数据,filter…暂时看不出这里的作用,columns
控制展示的每一列,其中最后一项的 CButtonColumn
向我们展示了三个按钮,分别是 查看 更新 和 删除。 接下来我们一点点地改造.

同时在db组件中加入
复制代码
代码如下:’enableProfiling’=>true,
同时在这种情况下,可以用CDbConnection::getStats()
查看执行了多少个语句,用了多少时间

用 CGridView 展示我们真正要的数据形式:
很多时候,数据库里的东西不适合直接展示给用户看,需要我们进行一定的处理之后才适合阅读。但在这里不经修改的话
CGridView
只会把数据库的值原封不动地呈现,所以,我们应该在相应的字段进行修改。比如
is_delete 字段,数据库里存放的是 0 和
1,但是在这里阅读就不太好了,我们应该改成 1 展示 ‘是’ ,0展示
‘否’。看看下面的代码,我们用了一个 array,两个键分别是 name 和
value,name 对应的要填写该 model 拥有的字段,而 value
是你想展示的数据,这里可以写成一个 php
语句,作为可以执行的代码。看到这里,是不是觉得对这个 value
我们可以做很多东西?有的同学可能要问,如果我想执行的代码很长,难道都写在
value
里面?。。。我说同学,你不会在其他地方写成一个函数然后在这里调用它吗??

如何知道某一个程序段运行需要的时间

php $this-widget('zii.widgets.grid.CGridView', array( 'id'='post-grid', 'dataProvider'=$model-search(), 'filter'=$model, 'columns'=array( 'post_id', 'title', 'content', 'tags', 'status', 'create_time', 'update_time', 'author_id', 'is_delete', array( 'name'='is_delete', 'value'='is_delete"是":"否"' //value 是可以执行 php 语句的哦 ) array( 'class'='CButtonColumn', ), ), ));  

配置好CProfileLogRoute后,在需要测试的地方加上

除此之外,还有一些常用的选项,都可以在 array
里面填写,下面是比较常见的使用方式:

Yii::beginProfile('blockID');
//程序段
Yii::endProfile('blockID');
array( 'name'='is_delete', 'value'='is_delete"是":"否"' //value 是可以执行 php 语句的哦 'filter' = array(0='否',1='是'), //自己定义搜索过滤的方式,这里为 是 和 否 的下拉菜单 'htmlOptions'=array('class'='delete'), //可以定义 html 选项,这里是定义了带一个 delete 的类 ), 

‘enableParamLogging’=>true,的作用是?

上面我们用 name 的话那是 model
里原来就有的字段,如果我们想展示自己定义的新内容呢,用 header :

在日志的bind的参数后边跟数的值

array( 'header'='备注', 'value'= 'display your data' ), 

如何在页面底部显示所有的db相关的日志

添加 CCheckBoxColumn :
有时也许我们会需要一个复选框,来对每一行进行选择,这时,我们可以增加一列,用
CCheckBoxColumn 类:

同上,配置log组件的routes中加入

php $this-widget('zii.widgets.grid.CGridView', array( 'id'='post-grid', 'dataProvider'=$model-search(), 'filter'=$model, 'columns'=array( array( 'selectableRows' = 2, //允许多选,改为 0 时代表不允许修改,1 的话为单选 'class' = 'CCheckBoxColumn',//复选框 'headerHtmlOptions' = array('width'='18px'),//头部的 html 选项 'checkBoxHtmlOptions' = array('name' = 'myname','class'='myclass'), //复选框的 html 选项 ), 'post_id', 'title', 'content', 'tags', 'status', 'create_time', 'update_time', 'author_id', 'is_delete', array( 'name'='is_delete', 'value'='is_delete"是":"否"', //value 是可以执行 php 语句的哦 'filter' = array(0='否',1='是'), //自己定义搜索过滤的方式,这里为 是 和 否 的下拉菜单 'htmlOptions'=array('class'='delete'), //可以定义 html 选项,这里是定义了带一个 delete 的类 ), array( 'class'='CButtonColumn', ), ), )); 
array(
'class'=>'CWebLogRoute',
'levels'=>'trace, info, error, warning',
'categories' => 'system.db.*',
//'showInFireBug' => true, 将在firebug中显示日志
),

修改ButtonColumn:
注意到列表每一项的最后三个小图标吗?不需要的话当然是直接删了,那要是只要其中某几个呢?可以加一个
template 参数:

把日志记录到数据库

array( 'class'='ButtonColumn', 'template'="{view} {update}", ), 
array(
'class'=>'CDbLogRoute',
'logTableName'=>'applog',
'connectionID'=>'db',
),

也可以自定义按钮:

运行时表applog会自动生成,如果不能生成,参照api自已建立

array( 'class'='ButtonColumn', 'template'="{view} {update} {print}", 'buttons'=array( 'print'=array( 'label'='打印', 'url'='Yii::app()-controller-createUrl("print", array("id"=$data-post_id))', 'options'=array("target"="_blank"), ), ), ), 

如何记录$_GET,$_SESSION等信息,在以上的routes中各个配置中加上 复制代码
代码如下:’filter’=>’CLogFilter’,
log配置中的level设置不对,可能会得不到日志信息

刷新时触发 Javascript: 如果你想在每次搜索之后触发一些 Javascript
,Yii 也提供了这个选项,你只要写成一个函数然后设置 afterAjaxUpdate
就好,记住这只是在 ajax 请求完成之后调用的,如果你想在 页面
一开始加载完成就调用的话需要另外加到页面的 Javascript

另外level,category的值可以随便写,

 $js = _JS_ function(){ alert('The ajax finish'); } _JS_; $this-widget('zii.widgets.grid.CGridView', array( 'id'='post-grid', 'dataProvider'=$model-search(), 'filter'=$model, 'afterAjaxUpdate'=$js, //看这里,ajax 之后调用的 javascript 在这里.... 'columns'=array( array( 'selectableRows' = 2, //允许多选,改为 0 时代表不允许修改,1 的话为单选 'class' = 'CCheckBoxColumn',//复选框 'headerHtmlOptions' = array('width'='18px'), 'checkBoxHtmlOptions' = array('name' = 'myname','class'='myclass'), ), .... 

只要在用yii::Log(“”,”自定义level”,”自定义的category”)时对应起来即可

添加 关联表 相关字段的搜索: 先说一句,我们在这里只谈 ”一对多“
的关联搜索,首先,不要忘了我们的数据库,忘记的同学请戳这里:这里,可以看到在
tbl_post 中是有一个外键关联到 tbl_user
表的,用以查找作者的相关信息。建了数据库之后,看看我们生成的 Yii 代码的
POST 的 Model, 里面的 realtion 如下:

如何记录更详细的信息,能记录stack,在入口文件中加上
复制代码
代码如下:define(‘YII_TRACE_LEVEL’,10);数字越大,记当的越详细,结果如下
[15:31:57.226][trace][system.db.CDbCommand] Querying SQL: SHOW
COLUMNS FROM `Bangdan` in
E:APMServ5.2.6wwwhtdocsdayouhui.comprotectedmodelsBangdan.php (21)
in E:APMServ5.2.6wwwhtdocsdayouhui.comprotectedcomponentsHotBangdan.php
(21) in E:APMServ5.2.6

/** * @return array relational rules. */ public function relations() { // NOTE: you may need to adjust the relation name and the related // class name for the relations automatically generated below. return array( 'comments' = array(self::HAS_MANY, 'Comment', 'post_id'), 'author' = array(self::BELONGS_TO, 'User', 'author_id'), ); } 

如果在调试时,终止程序运行且看到日志,不能用die及exit;

可以看到 POST 和 USER 表可以通过 author 键进行访问,例如:
$model-author-nickname,而且 这里是 BELONGS_TO 关系。
说了这么多,我们的需求究竟是什么?….

用application::end,即Yii::app()->end(),其会触发onEndRequest事件,日志就是在这个事件中记录的

产品经理推了推眼镜:”我们要在日志的后台管理界面加一个功能,可以通过作者名称搜索到相应的文章。这个比较急,今晚就要完成。“

activeRecord几个占位方法重写的注意点

淡定淡定,不就是改需求吗。忽略进度要求,我们研究一下究竟要做什么。
其实很简单的,不就是在 POST 的 admin
界面中增加一列作者名称,然后可以通过作者名的 模糊搜索
去找到对应日志吗?看看代码,要是通过 作者 id
去搜索不就简单了吗?不过这样确实不太友好…如果是展示作者名字而已不也是很简单吗?加一个
header 然后 value 是 $data-author-username,
问题是这样只能展示,不能进行搜索…哎,好苦恼。
淡定淡定,不就是多个搜索吗?来,让我告诉你怎么做。

必须带boolean返回值

首先,我们进入 POST 的 model,在一开始的地方添加一个属性:

如何发布一个资源文件并引用

class Post extends CActiveRecord { public $name; //添加一个 public 属性,代表作者名 然后改一下 Model 里面 search 的代码,改动部分都已经加了注释:public function search() { // @todo Please modify the following code to remove attributes that should not be searched. $criteria=new CDbCriteria; $criteria-with = array('author'); //添加了和 author 的渴求式加载 $criteria-compare('post_id',$this-post_id); $criteria-compare('title',$this-title,true); $criteria-compare('content',$this-content,true); $criteria-compare('tags',$this-tags,true); $criteria-compare('status',$this-status); $criteria-compare('create_time',$this-create_time); $criteria-compare('update_time',$this-update_time); $criteria-compare('author_id',$this-author_id); //这里添加了一个 compare, username 是 User 表的字段,$this-name 是我们添加的属性,true 为模糊搜索 $criteria-compare('username',$this-name,true); return new CActiveDataProvider($this, array( 'criteria'=$criteria, )); } 
$css=Yii::app()->getAssetManager()->publish(dirname(apizl_file)."/aa.css");
yii::app()->clientScript->registerCssFIle($css);

然后在 view 里面,就是 post 文件夹的 admin.php ,CGridView
改为下面代码:

如果改变activelable中默认的标题

php $this-widget('zii.widgets.grid.CGridView', array( 'id'='post-grid', 'dataProvider'=$model-search(), 'filter'=$model, 'columns'=array( 'post_id', 'title', 'content', 'tags', 'status', 'create_time', 'update_time', 'author_id', /*下面就是添加的代码啊*/ array( 'name'='作者名称', 'value'='$data-author-username', //定义展示的 value 值 'filter'=CHtml::activeTextField($model,'name'), //添加搜索 filter ), array( 'class'='CButtonColumn', ), ), ));  

重写方法attributeLabels

你是不是发现现在有了搜索框但是不起作用呢?哈哈,所以我们说文章要坚持看到最后。我们要做的最后一步,就是在
rule 里面,把 name 属性加入到安全搜索字段中,要不然会被 Yii
认为是不安全字段而过滤掉的。看,就在下面函数的最后一行,safe 前面多了个
name ….

过滤不良代码:

public function rules() { // NOTE: you should only define rules for those attributes that // will receive user inputs. return array( array('title, content, status, author_id', 'required'), array('status, create_time, update_time, author_id', 'numerical', 'integerOnly'=true), array('title', 'length', 'max'=128), array('tags', 'safe'), // The following rule is used by search(). // @todo Please remove those attributes that should not be searched. array('post_id, title, content, tags, status, create_time, update_time, author_id, name', 'safe', 'on'='search'), ); } 
$purifier=new CHtmlPurifier;
$purifier->options=array("HTML.Allowed"=>"div");
$content=$purifier->purify($content);

或者

beginWidget('CHtmlPurifier'); ?>
...display user-entered content here...
endWidget(); ?>

如何防止重复提交?

提交后
复制代码
代码如下:Ccontroler->refresh();
如何在成功后显示一个提示,用户刷新页时去掉提示

Cwebuser->setFlash();
getFlash();

如何防止重复提交, 并在提交成功后给出提示?

控制器中:

Yii::app()->user->setFlash('submit','thanks');
$this->refresh();

view中:

if(Yii::app()->user->hasFlash('submit')){
echo Yii::app()->user->getFlash('submit');
}

一般我们是跳转到列表页,或用redirect跳到编辑页,就不需要了,如果还是要显示当前页

以上就有用了,比如在当前时显示,编辑或添加新的记录

如何分页

itemCount总记录条数
CPagination代表分页信息,有多少页,每页几条记录等
CLinkPager生成分页的代码,自定义css可以给属性cssFile一个值

$criteria=new CDbCriteria();
$pages=new CPagination("数据库中的总记录数");
$pages->pageSize=2;
$pages->applyLimit($criteria);//给$criteria->limit offset等符值
$posts=Post::model()->findAll($criteria);
$this->widget('CLinkPager',array('pages'=>$pages));

列表如何排序

$criteria=new CDbCriteria();
$sort = new CSort('Post');
$sort->defaultOrder=" status asc";
$sort->applyOrder($criteria);
$posts=Post::model()->findAll($criteria);

应用时用:
复制代码
代码如下:$sort->link(‘字段名’)
实际是生成一个带参数的url,然后在在applyOrder时应用这些参数修改$criteria,得到相应的查寻结果

如何生成并验证验证码:

基本用法
复制代码 代码如下:widget(‘CCaptcha’);
?>具体参数查手册
原理CCaptcha这个widget会在run时调用当前控制器的$captchaAction=’captcha’方法,这个方法指到一个类CCaptchaAction

其会生成验证码图象,并记入到session中

如何显示静态页

重写actions:

'help'=>array(
'class'=>'CViewAction',
'basePath'=>'help', //指定目录名
'defaultView'=>'default',
'viewParam'=>'help' //get参数
),

假定当前控制器是post

那么可以能过/post/help/help/content访问help目录下的content.php

可以建立子目录比如help/reigterhelp/content.那可以通过/post/help/help/registerhelp.content来访问

用CViewAction的好处时,可以与其它的view共享layout

关于没有权限访问跳转的url相关

当没有权限时调用CAccessControlFilter类中的accessDenied,其调用CwebUser中的loginRequired(),记录当前的returnurl后跳转到
CWebUser配置中的loginurl,在此处登陆后,可以通过redirect跳转到复制代码
代码如下:returnurl(Yii::app()->request->redirect(Yii::app()->user->returnUrl);)
当强制显示登陆表单,比如判断用户是guest就一直列出登陆表单,不会调用loginRequired,
就得不到returnurl,这时候想跳回去,参见cookbook上相关贴子

registerCoreScript

在framework/web/js/package.php中列出的才是多对多关联条件

$criteria->addInCondition("categorys.id",$in);
$criteria->addSearchCondition('Shop.name',$keyword);
$shops=Shop::model()->with(array("categorys"=>array('together'=>true)))->findAll($criteria);

同时要在Shop模型中加入alias=”categorys”
,另外together=true放在模型的关联中也可

YII中的RBAC权限,用数据库存item,

在system/web/auth下找到相应的sql导放到数据库中

配置

'authManager' => array(
'class' => 'CDbAuthManager',
'connectionID' => 'db',
),

如果在sql中导入的三个表的表名不是默认的,需要在这上边的配置中配置,具体的看api

$auth=Yii::app()->authManager;
//$auth->createOperation("post",'postpost');
//$auth->createTask("post","posts");
$auth->createRole("post","post");
auth->assign("post",'demo');
if(Yii::app()->user->checkAccess("post")){
echo "yes";
else{
echo "no";
}

这种情况下三者是一样的

如何获得上一页的url以返回
复制代码
代码如下:Yii::app()->request->urlReferrer;
accessControl 是Ccontroller中内置的过滤方法,其它的还有ajaxOnly
postOnly
CMaskedTextField此组件用于限制用户的输入,对应的jquery插件

在一对多,多对多查询时,the eager loading
联合所有的表生成一条语句,如果主表有limit的查询选项,那么他将单独执行,然后再执行与关联表有关的语句,返回相关表的数据对象,这就是为什么在做大优惠时,以中间表为查询条件出错的原因,

解决办法

with()返回 CActiveFinder对象,其方法together(),既使主表中有LIMIT/OFFSET
也是返回一条sql;

多对多查询时,分页有时候页中显示的条数不正确,因为有重复的项,加上复制代码 代码如下:$criteria->group =
true即可
模型的rules中,验证某个字段不能重复,array(‘name’, ‘unique’,’message’
=> ‘有重复的名子’),

CStatePersister是yii的核心组件,提供了基于文件的数据保存方式,可以不在同的请求中使用

COutputCache
即是一个组件,又是一个filter,前者的时候用于在view中缓存内容,后者的时候用于在controller中缓存

就是说片段缓存,是把COutputCache当一个widget来用,页面缓存把COutputCache当作一个filter来用动态缓存,用CController的一个方法
renderDynamic($callback);

COutputCache几个属性,duration,dependency

另外还有几个,可以通称为Variation, 有什么作用呢?

在beginCache是需要手工指定一个id,Variation的作有就是自动给生成这个id
在布署模式的时候,有错误不会有stack样的提示,会显示一个errorxxx的错误

如何在程序有错的时候跳到指定的action

在components中设置

'errorHandler'=>array(
'errorAction'=>'site/error',
),

在此action中可以能过复制代码
代码如下:Yii::app()->errorHandler->error获得错误信息
把字符串分解成数组,并去掉空值
复制代码
代码如下:preg_split(‘/s*,s*/’,’this , is , , a
test’,-1,PREG_SPLIT_NO_EMPTY )
CActiveRecord::exits();判断有没有这样的记录,一般用于添加时,判断某字段有没有重复
CActiveDataProvider 一个基于ActiveRecord的数据提供源

常用的用法

$dataProvider=new CActiveDataProvider('Post', array(
'criteria'=>array(),
'pagination'=>array(),
'sort'=>array(),  
));

上如

'sort'=>array(
'defaultOrder'=>'status, update_time DESC',
),

ClistView同上结合使用,其中的_view中可以用一个$data的变量,代表当前的model数据
如果dataProvider中的pagination,sort设为false,则CliveView中对应的部分也无法使用

$this->widget('zii.widgets.ClistView',array(
'dataProvider' => $dataprovider,
'itemView' => '_view',
'template' => '{items}{sorter}{pager}',
'sortableAttributes' => array(
),
));

CGridView的使用也结合$dataprovider,
用的时候主要是对columns的配置,主要有
CDataColumn, CLinkColumn, CButtonColumn and
CCheckBoxColumn.具体用法看api
总的说来CgridView没有ClistView灵活

插入meta信息

Yii::app()->clientScript->registerMetaTag('keywords','关键字');
Yii::app()->clientScript->registerMetaTag('description','一些描述');

CMap::mergeArray() 比array_merge更智能的合并数组,yii中配置的合并用这个

CClipWidget 通过ob_start
ob_getconent生成一段不显示的内容,可以能过CController::clips访问,如
复制代码
代码如下:$this->beginWidget(‘CClipWidget’,array(‘id’=>’name’,’renderClip’=>true));
可以通过$this->clips[‘name’]来显示,其中的renderClip如果为false,则在当前位置不显示内容

如果在Model一次验证多个属于,显示不同的内容?如下 
[后来发现这个不起作有]

return array(
array('title, content', 'required',
'message'=>'Please enter a value for {attribute}.'),
// ... other rules
);

获得服务器时间
复制代码
代码如下:$_SERVER[‘REQUEST_TIME’]
维护程序时,这样子所有的请求转发到一个地方
复制代码
代码如下:’catchAllRequest’=>array(‘site/all’),
根据二级域名缓存

array(
'COutputCache + search',
'duration' => 120,
'varyByParam' => array('q','page'),
'varyByExpression' => "app()->request->hostInfo",
),

有多个分站时,同步登陆,基于cookie

'user'=>array(
'identityCookie'=>array('domain'=>'.dayouhui.com'),
'allowAutoLogin' => true,
)

如果是基本于session

'session' => array(  
'cookieParams' => array('domain' => '.dayouhui', 'lifetime' => 0),
'timeout' => 3600,
),

如何使用theme

在main.php中配置
复制代码
代码如下:’theme’=>’classic’,
如何得到当前使用的主题
复制代码 代码如下:Yii::app()->theme
得到名子
复制代码
代码如下:Yii::app()->theme->name;
themes文件夹和protected是同级的,其下边某个theme的目录结果同protected/views下一样

关于skin

用theme改变view的外观,skin是用来改变widgets的外观的
skin是健值对用于初始化一个widget的属性
要对widget使用skin,需要做以下几步

1:配置

'widgetFactory'=>array(
'enableSkin'=>true,
),

2:在views下建立skins目录

3:在skins目录下建立与Widget名子一样的php文件,返回数组,即能用于widget的初始配置

4:在php文件中,如果有defautl的配置,会先找这个skin

5:如果应用了theme,程序会先去对应的theme目录下的skins中找配置文件

6:如果只是想给widget统一一个skin,建议用Customizing Widgets Globally
如果防止post跨站攻击

'request'=>array(
'enableCsrfValidation'=>true,
),

这时候生成的表单要用CHtml::form(),其会写一段代码在cookie中

防止Cookie攻击

'request'=>array(
'enableCookieValidation'=>true,
),

同时生成与得到cookie是要用 CHttpCookie

如何让表单验证不通过的提示为中文

在main.php的配置中加上
复制代码 代码如下:’language’ =>
‘zh_CN’,
如何实现仿google的自动完成功能

widget('CAutoComplete', array(
'name'=>'xxx',
'url'=>array('suggestTags'),
'multiple'=>false,
'htmlOptions'=>array('size'=>50),
));
?>

然后在url指定的地址中的方法中如下输出,即可

echo “anbnc”
//CGridView详解

这东西在后台比较有用,能加速开发的速度,值得一看

CGridView用表格的方式显示数据项

每一行代表一个数据项,一列通常代表数据项的一个属性
CGridView支持排序和分页,可以用ajax或普通的方式
CgridView必序和data provider一起使用

最简单的用法

$dataprovider = new CActiveDataProvider('Post');
$this->widget('zii.widgets.grid.CGridView',array(
'dataProvider'=>$dataprovider,
));

这会用表格的方式显示每一条数据项,每一列是Post的一个属性

在显示中带了分页和排序

我们可以自定义CgridView::columns属性,以自定义表格列的显示方式

这个cloumns如何配置呢?

其是一个数组,每一个数组元素对应着一列的配置,可以是字符串或数组

1、如果是字符串,格式是name:type:header
后两者是可选的,根据这三个值,创建一个CdatColumn实例

其中type参见CFormatter

2、如果是数组,其可以实例化CDdataColumn、ClinkColumn,CButtonColumn,CCheckBoxColumn实例,具体实例化哪个由数组中的class指定,默认是CDataColumn

2.1,如果复制代码
代码如下:class=>’CDataCloumn’
则可以指定name或者value,如果指定以value优先

用CDataColumn时如何以关联表的数据序列?

代码如下:表示可以post关联的author中的username排序列

$dataprovider = new CActiveDataProvider('Post',array(
'criteria'=>array(
'with'=>'author',
),
'sort'=>array(
'attributes'=>array(
'title','create_time',
'author_id'=>array('asc'=>'author.username asc','desc'=>'author.username desc','label'=>'作者')
)
),
));
$this->widget('zii.widgets.grid.CGridView',array(
'dataProvider'=>$dataprovider,
'columns'=>array(
'title',
'create_time',
array('name'=>'author_id','value'=>'$data->author->username'),
),
));

另外CDataColumn还有一个filter属性,如果是空,那么生成一个textfield,如果是数组(键值),则生成一个dropDownlist在当前列的上部,供搜索

2.2:如果class=>”CLinkColumn”
复制代码
代码如下:array(‘class’=>’CLinkColumn’,’label’=>’查看用户’,’url’=>Yii::app()->createURL(‘user/edit’))

则生成一个连接

2.3:如果class=”CCheckBoxColumn”
复制代码
代码如下:array(‘class’=>’CCheckBoxColumn’,’name’=>’title’,’id’=>’select’),

可以生成一个checkbox供选择,且只能选一个

可以配置CGridView::selectableRows 如果是0,则不能选,如果
1,只选一个如果是2或其它值,则可以选多个

代码如下:

$this->widget('zii.widgets.grid.CGridView',array(
'dataProvider'=>$dataprovider,
'selectableRows'=>2,
'columns'=>array(
array('class'=>'CCheckBoxColumn','name'=>'title','id'=>'select'),  
),

2.3:如果class=”CButtonColumn”

array(
'class'=>'CButtonColumn',
'updateButtonUrl'=>'Yii::app()->createUrl("post/edit",array("id"=>$data->id));',
),

修改updateButtonUrl为编辑贴子

如何用gridview生成一个代搜索的管理列表

1、在Model的rules 设定可以搜索的属性
复制代码 代码如下:array(‘title, status,
create_time’, ‘safe’, ‘on’=>’search’),
2、在Model中,添加搜索时的方法

public function search()
{
$criteria=new CDbCriteria;
$criteria->compare('title',$this->title,true);
$criteria->compare('status',$this->status);
$criteria->compare('create_time',$this->create_time);
return new CActiveDataProvider('Post', array(
'criteria'=>$criteria,
'sort'=>array(
'defaultOrder'=>'status, update_time DESC',
),
));
}

3、在Controler中,写接受搜索用到的表单的值的方法

public function actionAdmin()
{
$model=new Post('search');
if(isset($_GET['Post']))
$model->attributes=$_GET['Post'];
$this->render('admin',array(
'model'=>$model,
));
}

4、在view中用CGridView显示

设置好

widget('zii.widgets.grid.CGridView', array(
'dataProvider'=>$model->search(),
'filter'=>$model,
'columns'=>array(
),
));
?>

以上代码大部分是yii自动生成的,只要做少量修改即可

有时候会出现,搜索后页面为空的清况,原因可能是

layout/main.php中

echo $content外层无div,就是说main.php中必须有一个div包含$content

其用列表的形式显示数据,不象CGridView一样,用表格显示数据,CListView用一个
view模板来显示每一条数据

其支持排序与分页

常用的代码如下

array(
'pageSize'=>2
),
));
$this->widget('zii.widgets.CListView',array(
'dataProvider'=>$dataProvider,
'itemView'=>'_view',
'template'=>' {summary} {items} {pager}{sorter}',
'sortableAttributes'=>array(
'title',
'create_time'=>'Post Time',
),
));

快速生成表单,支持ajax验证,对于比较复杂的验下最好是自己生成表单,写验证方法
常用代码,在Controller中

public function actionForm()
{
$post = new Post();
if(isset($_POST['ajax']) && $_POST['ajax']==='post'){
echo CActiveForm::validate($post);
Yii::app()->end();
}
if(isset($_POST['Post'])){
$post->attributes = $_POST['Post'];
if($post->save()){
echo '存成功了';
}
}
$this->render('form',array('post'=>$post));
}

在view中

beginWidget('CActiveForm',array(
'id'=>'post',//这里与Controller中的ajax对应
'enableAjaxValidation'=>true,
));
?>

labelEx($post,'title');?>
textField($post,'title')?>
error($post,'title'); ?>

error一定要写上,要不不会触发ajax验证

labelEx($post,'content');?>
textField($post,'content')?>
isNewRecord ? 'Create' : 'Save'); ?>
endWidget(); ?>
//CBreadcrumbs常用代码
widget('zii.widgets.CBreadcrumbs', array(
'links'=>$this->breadcrumbs,
'homeLink'=>'shouye',
'separator'=>'>>>'
)); ?>

其中breadcrumbs中Controller中的一个属性,如果要出现导航,就要在view中给此属性附值

生成的html如下

shouye
>>>Managde Posts>>>
b>>>c<>

所以如果网站用到导航的时候,美工最好把导航代码定义如上

//CDetailView  用在仅仅是为了查看数据时,还是比较有用的,比如用在后台

如何在提交后显示一段提示

在控制器中

if(isset($_POST['name'])){
Yii::app()->user->setFlash('success','you are success');
$this->refresh();
}

在view中

if (Yii::app()->user->hasFlash('success')){
echo 're is'.Yii::app()->user->getFlash('success');
}else{
echo 'no';
}

如何得到当前域名:
复制代码
代码如下:app()->request->hostInfo
activeDropDownList,给出提示,并有值

array('empty'=>array(0=>'选择分组')

验证码如何生成及验证:

Controller中:

public function actions()
{
return array(
'captcha'=>array(
'class'=>'CCaptchaAction',
'backColor'=>0xFFFFFF,
'maxLength'=>4,
'minLength'=>4,
),
);
}

View中

widget('CCaptcha',array(
'captchaAction' => '/site/captcha',
'showRefreshButton' => false,
'clickableImage' => true,
'imageOptions' => array('align'=>'top', 'title'=>'重新获取'),
));
?>

Model中

array('verifyCode', 'captcha', 'captchaAction'=>'site/captcha', 'message' => '输入的验证码不正确'),
set_time_limit(0);//禁止角本超时

如何想把手工的东西记录的数据库

main.php中配置log

array(
'class'=>'CDbLogRoute',
'levels'=>'info',
'logTableName'=>'Log',
'connectionID'=>'db',
),

应用时
复制代码
代码如下:Yii::log(‘信息’,’info’);
deleteAllByAttributes(array(“phone”=>$phones)直接接受一个数组,可以删除数组中符合条件的记录

YII_BLOG STUDY重新看了一遍yii blog,有些记录会与上边的重复

YII:Trace()
在debug模式是才记录信息,同时在main.php中的Log中的配置中的levels中要有trace,至于记录多少

栈由index.php中的YII_TRACE_LEVEL决定

配置Gii

'modules'=>array(
'gii'=>array(
'class'=>'system.gii.GiiModule',
'password'=>'123',
),
),

获得客户端IP

if($_SERVER['HTTP_CLIENT_IP']){
$ip = $_SERVER['HTTP_CLIENT_IP'];
}elseif($_SERVER['HTTP_X_FORWARDED_FOR']){
$ip = $_SERVER['HTTP_X_FORWARDED_FOR'];
}else{
$ip = $_SERVER['REMOTE_ADDR'];
}

CActiveForm还是比较强大的,建议在以后的项目中form都用这个来实现

layout/中的视图是可以继承的
复制代码
代码如下:beginContent(‘/layouts/main’); ?>
然后在中间出现$content即可
复制代码 代码如下:endContent(); ?>
create,update最好是分开放在两个action中,共用一个form,中间可以加一层view,以在头尾显示不同的东西

成段的完成一个功能的代码尽量拿出来放到一个方法中

$this->beginWidget('CMarkdown', array('purifyOutput'=>true));
echo $data->content;
$this->endWidget();

linkButton,在删除时需要用js提示,可以看下这此组件中的comfirm

而且他们的提交方式都是post,是因为在jquery.yii.js写死了

具体的以在源文件中低部找到那段js中的ajaxsubmit,所在的js看下

filter是在执行action之前或之后执行的一段代码,要应用filters必须得写

CController::filters()方法

为什么在filters方法写上

return array(
'accessControl', // perform access control for CRUD operations
);

能进行crud验证呢?

accessController是CContronller内置的filter,其调用
accessRules,得到验证规定,所以也要重写对应的accessRules,返回一个验证规则的数组成部分

if the application uses modules,
a root alias is also predefined for each module ID and refers to the
base path of the corresponding module

如:echo YiiBase::getPathOfAlias(‘bbs’);得到module bbs的路径

关于CUrlManager

复制代码 代码如下:’模式’=>’route’
matchValue是指,对于一个url规则,正常情况下是只看参数的名子是否一样就应用规则

如果matchValue=true,则也要看值

如,规则

'index-/'=>array("book/index",'matchValue'=>false),
$this->createUrl('book/index', array('id'=>'abcd'));

可以应用以上规则的,

如果规则中的matchValue=true,则就不能应用了

XSS又叫CSS (Cross Site Script) ,跨站脚本攻击。

它指的是恶意攻击者往Web页面里插入恶意html代码,当用户浏览该页之时,
嵌入其中Web里面的html代码会被执行

renderPartial()
render()

后者会把需要的js,css等嵌入

前者可以通过把最后一个参数设置成true完成一样的功能

addInCondition 不用考虑数组是空的情况yii会自动处理

如何得到当前url?
复制代码
代码如下:Yii::app()->request->url;
ctype_开始的几个函数,用于检查字任串是不是符合要求,代替了简单的正则表达式
CController中的setPageState可以保存同一页中的POST的表单状态

如何通过BEhavior修改CActiveRecord?

写类文件继承自

class LLog extends CActiveRecordBehavior{
public function beforeDelete($event){
$model = get_class($this->Owner);
//做要做的事,比如日志或修改模型字段内容
}
}

然后修改模型文件

public function behaviors()
{
return array(
// Classname => path to Class
'LLog'=>'application.behavior.LLog',
);
}

如何在应用程序处理请求之前执行一段操作?

在main.php中配置
复制代码 代码如下:’onBeginRequest’ =>
‘function’
当然这个function方法要存在

也可以写在放口文件index.php中,代码改成如下

$app = Yii::createWebApplication($config);
$app->onbeginRequest = 'begin';
$app->run();
function begin(){
echo 'yyyyydddyyyyyy';
}

为什么在CActiveRecordBehavior中用

beforesave就可以代表了事件onBeforeSave

注意基为中最上边的events方法中返回的对应关系
复制代码
代码如下:’onBeforeSave’=>’beforeSave’
在调用attacth(CBehavior中)的时候,
复制代码
代码如下:$owner->attachEventHandler($event,array($this,$handler));
就指定了事件onBeforeSave的处理函数是用本类中的beforeSave
YII中的CComponent,CEvent与Behavior及CActiveRecordBehavior个人理解

这一块教程少,今天个人理解了下,写了个小例子,有助于理解

完成如下功能,一个JTool类,继承CComponent,当其长度改变时,调用事件,输出”change
me”.

JTool.php在protected/components 下

_width ? $this->_width : 1; 
}
public function setWidth($width){
if($this->hasEventHandler('onChange')){
$this->onChange(new CEvent());
}
$this->_width = $width;
}
public function onChange($event){
$this->raiseEvent('onChange', $event);
}
}

OK,功能已经实现了,找个控制器,执行

$j = new JTool();
$j->onChange = "showChange"; //给事件绑定handle showChange
$j->width = 100; //调用setWidth,解发绑定的事件showChange
function showChange(){
echo 'changed me';
}

现在我们想给JTool添加一个功能,返回长度的100倍,我们可以继承JTool.php写一个方法

class JToolSub extends JTool{
public function get100width(){
return $this->width*100;
}
}

OK,功能实现了,这个执行就简单了new JToolSub调用方法即可

上边的这两种办法,就是仅完成功能,下边演示Behavior及events来实现

如何用Behavior来实现上边的增加一个方法,返回长度的100倍的功能呢?
写类JBe

JBe.php在protected/behavior 下

class JBe extends CBehavior{
public function get100width(){
return $this->Owner->width*100;
}
}

OK,功能已经实现了,找个控制器,执行

$j = new JTool();
$j->attachBehavior('JBe', 'application.behavior.JBe');
echo $j->get100width();

如何用Behavior实现JTool中的长度改变时,调用一个事件的功能呢?
写类JBe

class JBe extends CBehavior{
public function events(){
return array_merge(parent::events(),array(
'onChange'=>'change',
));
}
public function change(){
echo 'changed';
}
public function get100width(){
return $this->Owner->width*100;
}
}

OK,功能实现随便找个控制器,执行

$j = new JTool();
$j->attachBehavior('JBe', 'application.behavior.JBe');
$j->width = 100;

这里的要点是events方法

返回的数组array(‘onChange’=>’change’)定义了事件(event)和对应的事件处理方法(event
hander)

事件是是Compents(JTool中)定义的,即JTool中的onChange

处理方法同由Behavior(JBe中)类定义的,即JBe中的change

这样子再看CActiveRecordBehavior,其是绑定给CActiveRecord
这个组件的,绑定方法重写behaviors()

CActiveRecordBehavior中的events() 方法返回事件及事处理函数的对应,如:
复制代码
代码如下:’onBeforeSave’=>’beforeSave’
即组件CActiveRecord中的onBeforeSave这个事件对应的处理函数是CActiveRecordBehavior中的beforeSave方法

这样子CActiveRecord在调用save()时,触发事件onBeforeSave,调用CActiveRecordBehavior对应的处理函数beforeSave

我们只要写一个CActiveRecordBehavior的子类,重写其中的beforeSave,执行一些操作,然后给CActiveRecord绑定即可

如果你自己有个目录下有些类或文件常用,可以在main.php的最上边定义一个路径别名
复制代码
代码如下:Yii::setPathOfAlias(‘local’,’path/to/local-folder’);
如果是多个可以在main.php中的array中加一个配置

'aliases'=>array(
'local'=>'path/to/local/'
),

如何得到proteced目录的物理路径? 复制代码
代码如下:YII::app()->basePath;
widget是发布资源

$url = Yii::app()->getAssetManager()->publish(Yii::getPathOfAlias('application.components.homeuserlived'));
cs()->registerCoreScript('jquery');
cs()->registerScriptFile($url.'/location.js' ,CClientScript::POS_HEAD);
cs()->registerScriptFile($url.'/YLChinaArea.js' ,CClientScript::POS_HEAD);
cs()->registerCssFile($url.'/style.css');

如何写application component, 即在main.php可配置
复制代码 代码如下:”my”=>array(”)
可以通过Yii::app()->my来访问?

继承CApplicationComponent即可,并可以自带Behavior等

yii中读写session的两种方法

$session = Yii::app()->session;
$session['terry'] = 30;
var_dump($session['key']);
Yii::app()->user->setState('tom', '40');
var_dump(Yii::app()->user->getState('key', 'default'));

soap非yii教程,意思是不用yii框架的时候要对象提供webservice的写法

分两种WSDL模式,和非WSDL模式,先看后者

这个也比较简单,服务器端server.php:

";
$xml .= "".$age."";
return $xml;
}  
}
$soapS = new SoapServer(null,array('uri' => 'http://www.dayouhui.com'));
$soapS->setClass('Student');
$soapS->handle();
?>

客户端client.php:

"http://localhost/mysoap/index.php",'uri'=>'inadex.php'));
echo $soap->getInfo('a','b');

这样子即可

yii,Componnts那快,忘了,写了个小例子回忆了下
是写一个可以写在main.php中的Components并绑定行为,事件

class ExtWindow extends CApplicationComponent{
private $title = 'title';
public $oldtitle;
public function getTitle(){
return $this->title ? $this->title : 'old title
';
}
public function setTitle($title){
echo '=='.$this->oldtitle.'==';
$this->oldtitle = $this->title;
$this->title = $title;
if($this->hasEventHandler('onTitleChange')){
$event =new CEvent($this);
$this->raiseEvent('onTitleChange', $event);
}
}
//必须有这么个方法,其和raiseEent中的事件一样,具体看代码
public function onTitleChange($event){
}
}

'titleChange',
));
}
public function titleChange($event){
echo $event->sender->title;
echo 'event TitleChange is handled in Behavior
';
echo $this->owner->title;
}
public function titleOld(){
echo '
old title is is '.$this->owner->oldtitle;
}
}

main.php中的写法

'ExtWin'=>array(
'class' => 'ExtWindow',
'oldtitle'=>'我是旧的',
'behaviors'=>array('win'=>'application..behavior.Window')

一对多,多对多的关联时最后的参数 together说明

如果为false,分开查多个语句
如果为true,强制生成一个语句
如果没有设置,分页页生成多个语句,不分页时生成一个语句
),
多对多时,查询时,中间表的名子叫 (关联名_关联名)
with选项的作用是eager loading
together的作用是 要不要形成一个语句

当是一个sql语句是记录会有重复,这时候分页分出现相同的记录,加上group=>true即可,

只要弄明白了,你生成的sql是一条还是多条sql就明白在多对多查询时的结果了

两个表不是用主键关联
复制代码 代码如下:’user’ =>
array(self::BELONGS_TO, ‘OaskUser’, ”,’on’=>’name=userName’,
‘select’=>’TrueName’),
表带talbeprefix是,多对关联,键的写法
复制代码
代码如下:’categories’=>array(self::MANY_MANY,’Category’,'{{post_category}}(post_id,category_id)’),

希望本文所述对大家基于Yii框架的PHP程序设计有所帮助。

发表评论

电子邮件地址不会被公开。 必填项已用*标注