• Yii框架tips(转)


    yii的一些小的技巧

    http://www.yiichina.com/topic/151

    db组件 'schemaCachingDuration'=>3600, 为什么不起做用?
    需要开缓存


    如何在页面下边显示sql的查询时间
    在log组件的routes中加入

    array( 
    'class'=>'CProfileLogRoute', 
    'levels'=>'error, warning', 
    )

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


    如何知道某一个程序段运行需要的时间
    配置好CProfileLogRoute后,在需要测试的地方加上

    Yii::beginProfile('blockID'); 
    //程序段 
    Yii::endProfile('blockID');


    'enableParamLogging'=>true,的作用是?
    在日志的bind的参数后边跟数的值

    如何在页面底部显示所有的db相关的日志
    同上,配置log组件的routes中加入

    array( 
    'class'=>'CWebLogRoute', 
    'levels'=>'trace, info, error, warning', 
    'categories' => 'system.db.*', 
    //'showInFireBug' => true, 将在firebug中显示日志 
    ),

    把日志记录到数据库

    array( 
    'class'=>'CDbLogRoute', 
    'logTableName'=>'applog', 
    'connectionID'=>'db', 
    ),

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

    如何记录$_GET,$_SESSION等信息,在以上的routes中各个配置中加上

    'filter'=>'CLogFilter',

    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.6wwwhtdocsdayouhui.comprotectedmodelsBangdan.php (21)
    in E:APMServ5.2.6wwwhtdocsdayouhui.comprotectedcomponentsHotBangdan.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

    过滤不良代码

    $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,得到相应的查寻结果


    如何生成并验证验证码:
    基本用法
    <?php $this->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插件http://digitalbush.com/projects/masked-input-plugin/


    在一对多,多对多查询时,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,则在当前位置不显示内容


    获得服务器时间

    $_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 "a b c"

    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或其它值,则可以选多个
    代码如下:

    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显示
    设置好
    <?php $this->widget('zii.widgets.grid.CGridView', array(
    'dataProvider'=>$model->search(),
    'filter'=>$model,
    'columns'=>array(
    ),
    )); ?>
    以上代码大部分是yii自动生成的,只要做少量修改即可
    有时候会出现,搜索后页面为空的清况,原因可能是
    layout/main.php中
    echo $content外层无div,就是说main.php中必须有一个div包含$content
    //CListView详解
    其用列表的形式显示数据,不象CGridView一样,用表格显示数据,CListView用一个 view模板来显示每一条数据
    其支持排序与分页
    常用的代码如下
    <?php
    $dataProvider = new CActiveDataProvider('Post',array(
    'pagination'=>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',
    ),
    ));

    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>&gt;&gt;&gt;<span>Managde Posts</span>&gt;&gt;&gt;
    <span>b</span>&gt;&gt;&gt;<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',
    ),
    ),


    获得客户端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/中的视图是可以继承的
    <?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的处理函数是用本类中的beforeSave
    YII中的CComponent,CEvent与Behavior及CActiveRecordBehavior个人理解
    这一块教程少,今天个人理解了下,写了个小例子,有助于理解
    完成如下功能,一个JTool类,继承CComponent,当其长度改变时,调用事件,输出"change me".
    JTool.php在protected/components 下
    <?php
    class 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,解发绑定的事件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
    <?php
    ini_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){
    }
    }
    ===========================
    <?php
    class 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 loading
    together的作用是 要不要形成一个语句
    当是一个sql语句是记录会有重复,这时候分页分出现相同的记录,加上group=>true即可,
    只要弄明白了,你生成的sql是一条还是多条sql就明白在多对多查询时的结果了
    两个表不是用主键关联
    'user' => array(self::BELONGS_TO, 'OaskUser', '','on'=>'name=userName', 'select'=>'TrueName'),

    From: http://bbs.phpchina.com/blog-83109-193270.html

  • 相关阅读:
    【信仰充值中心】Pale Moon 29 正式版更新日志
    【亚伦博客】我们勇敢的审查者
    狐吧吧刊——遇见最好的浏览器(16期)
    【信仰充值中心】Firefox 97 后续更新一览
    JavaTPoint Web 中文教程【翻译完成】
    声纹识别
    设计模式
    WSL迁移安装目录
    openvSwitch概述
    进程间通信几种方式
  • 原文地址:https://www.cnblogs.com/imxiu/p/3416639.html
Copyright © 2020-2023  润新知