db组件 'schemaCachingDuration'=>3600, 为什么不起做用?需要开缓存
如何在页面下边显示sql的查询时间在log组件的routes中加入同时在db组件中加入
'enableProfiling'=>true,同时在这种情况下,可以用CDbConnection::getStats() 查看执行了多少个语句,用了多少时间 如何知道某一个程序段运行需要的时间配置好CProfileLogRoute后,在需要测试的地方加上如何在页面底部显示所有的db相关的日志
同上,配置log组件的routes中加入把日志记录到数据库
运行时表applog会自动生成,如果不能生成,参照api自已建立
如何记录$_GET,$_SESSION等信息,在以上的routes中各个配置中加上
log配置中的level设置不对,可能会得不到日志信息
另外level,category的值可以随便写,只要在用yii::Log("","自定义level","自定义的category")时对应起来即可如何记录更详细的信息,能记录stack?
在入口文件中加上
define('YII_TRACE_LEVEL',10);数字越大,记当的越详细,结果如下[15:31:57.226][trace][system.db.CDbCommand] Querying SQL: SHOW COLUMNS FROM `Bangdan` in E:\APMServ5.2.6\www\htdocs\dayouhui.com\protected\models\Bangdan.php (21)in E:\APMServ5.2.6\www\htdocs\dayouhui.com\protected\components\HotBangdan.php (21) in E:\APMServ5.2.6如果在调试时,终止程序运行且看到日志,不能用die及exit;用application::end,即Yii::app()->end(),其会触发onEndRequest事件,日志就是在这个事件中记录的如何发布一个资源文件并引用$css=Yii::app()->getAssetManager()->publish(dirname(__FILE__)."/aa.css");
yii::app()->clientScript->registerCssFIle($css);如果改变activelable中默认的标题重写方法attributeLabels过滤不良代码
或者
如何在成功后显示一个提示,用户刷新页时去掉提示
如何防止重复提交, 并在提交成功后给出提示?控制器中
view中
一般我们是跳转到列表页,或用redirect跳到编辑页,就不需要了,如果还是要显示当前页,以上就有用了,比如在当前时显示,编辑或添加新的记录
如何分页
itemCount总记录条数CPagination代表分页信息,有多少页,每页几条记录等CLinkPager生成分页的代码,自定义css可以给属性cssFile一个值应用时用
$sort->link('字段名')实际是生成一个带参数的url,然后在在applyOrder时应用这些参数修改$criteria,得到相应的查寻结果 如何生成并验证验证码:基本用法<?php $this->widget('CCaptcha'); ?> 具体参数查手册原理CCaptcha这个widget会在run时调用当前控制器的$captchaAction='captcha'方法,这个方法指到一个类CCaptchaAction其会生成验证码图象,并记入到session中如何显示静态页重写actions
假定当前控制器是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中列出的才是 多对多关联条件同时要在Shop模型中加入alias="categorys" ,另外together=true放在模型的关联中也可
YII中的RBAC权限,用数据库存item,在system/web/auth下找到相应的sql导放到数据库中配置'authManager' => array('class' => 'CDbAuthManager','connectionID' => 'db',),如果在sql中导入的三个表的表名不是默认的,需要在这上边的配置中配置,具体的看api这种情况下三者是一样的
如何获得上一页的url以返回动态缓存,用CController的一个方法 renderDynamic($callback);
COutputCache几个属性,duration,dependency 另外还有几个,可以通称为Variation, 有什么作用呢?在beginCache是需要手工指定一个id,Variation的作有就是自动给生成这个id 在布署模式的时候,有错误不会有stack样的提示,会显示一个errorxxx的错误 如何在程序有错的时候跳到指定的action在components中设置在此action中可以能过Yii::app()->errorHandler->error获得错误信息
把字符串分解成数组,并去掉空值CClipWidget 通过ob_start ob_getconent生成一段不显示的内容,可以能过CController::clips访问,如
根据二级域名缓存
基于cookie
如果防止post跨站攻击
防止Cookie攻击
如何让表单验证不驼过的提示为中文在main.php的配置中加上
然后在url指定的地址中的方法中如下输出,即可
echo "a\nb\nc"CGridView详解
这东西在后台比较有用,能加速开发的速度,值得一看CGridView用表格的方式显示数据项每一行代表一个数据项,一列通常代表数据项的一个属性CGridView支持排序和分页,可以用ajax或普通的方式CgridView必序和data provider一起使用最简单的用法用CDataColumn时如何以关联表的数据序列?代码如下:表示可以post关联的author中的username排序列
CActiveForm详解
快速生成表单,支持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中<?php$form = $this->beginWidget('CActiveForm',array('id'=>'post',//这里与Controller中的ajax对应'enableAjaxValidation'=>true,));?><?php echo CHtml::errorSummary($post); ?><?php echo $form->labelEx($post,'title');?><?php echo $form->textField($post,'title')?><?php echo $form->error($post,'title'); ?> error一定要写上,要不不会触发ajax验证<?php echo $form->labelEx($post,'content');?><?php echo $form->textField($post,'content')?><?php echo CHtml::submitButton($post->isNewRecord ? 'Create' : 'Save'); ?><?php $this->endWidget(); ?>//CBreadcrumbs常用代码<?php $this->widget('zii.widgets.CBreadcrumbs', array('links'=>$this->breadcrumbs,'homeLink'=>'<span><a href="http://abc.com">shouye</span>','separator'=>'>>>')); ?>其中breadcrumbs中Controller中的一个属性,如果要出现导航,就要在view中给此属性附值生成的html如下<div ><span><a href="http://abc.com">shouye</span>>>><span>Managde Posts</span>>>><span>b</span>>>><span>c</span>所以如果网站用到导航的时候,美工最好把导航代码定义如上//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=>'选择分组')<input type="submit" value="提交" /> 验证码如何生成及验证Controller中public function actions(){ return array('captcha'=>array('class'=>'CCaptchaAction','backColor'=>0xFFFFFF,'maxLength'=>4,'minLength'=>4,),);}View中<?php echo CHtml::activeTextField($user, 'verifyCode');?><?php $this->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',),), 获得客户端IPif($_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/中的视图是可以继承的<?php $this->beginContent('/layouts/main'); ?>然后在中间出现$content即可<?php $this->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-/<id:\d+>'=>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的处理函数是用本类中的beforeSaveYII中的CComponent,CEvent与Behavior及CActiveRecordBehavior个人理解这一块教程少,今天个人理解了下,写了个小例子,有助于理解完成如下功能,一个JTool类,继承CComponent,当其长度改变时,调用事件,输出"change me".JTool.php在protected/components 下<?phpclass JTool extends CComponent{ private $_width;public function getWidth(){ return $this->_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,解发绑定的事件showChangefunction 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倍的功能呢?写类JBeJBe.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中的长度改变时,调用一个事件的功能呢?写类JBeclass 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<?phpini_set('soap.wsdl_cache_enabled',0);class Student { public function getInfo($name,$age){ if($age == 20){ throw new SoapFault(-1, 'Cannot divide by zero!');}$xml = "<root><name>".$name."</name>";$xml .= "<age>".$age."</age></root>";return $xml;} }$soapS = new SoapServer(null,array('uri' => 'http://www.dayouhui.com'));$soapS->setClass('Student');$soapS->handle();?>客户端client.php<?php$soap = new SoapClient(null,array('location'=>"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<br />';}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){ }}===========================<?phpclass Window extends CBehavior{ public function events(){ return array_merge(parent::events(),array('onTitleChange'=>'titleChange',));}public function titleChange($event){ echo $event->sender->title;echo 'event TitleChange is handled in Behavior<br />';echo $this->owner->title;}public function titleOld(){ echo '<br />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 loadingtogether的作用是 要不要形成一个语句当是一个sql语句是记录会有重复,这时候分页分出现相同的记录,加上group=>true即可,只要弄明白了,你生成的sql是一条还是多条sql就明白在多对多查询时的结果了两个表不是用主键关联'user' => array(self::BELONGS_TO, 'OaskUser', '','on'=>'name=userName', 'select'=>'TrueName'),
From: http://home.phpchina.com/blog-83109-193270.html