wordpress自定义页面模板

对于页面的请求,wordpress默认调用page.php这个文件,不管我们添加多少个页面,它们的模板都是同一个——page.php。然而实事是,在很多情况下我们都想让某个页面和其它页面不太一样,那么怎么自定义这些页面呢?本文将给出答案。 条件判断 既然页面默认的模板文件是page.php文件,我们就可以通过在page.php文件根据不同的page-id来输出不同的模板样式。这种方法的基本框架如下
view plaincopy to clipboardprint?
[php] <?php switch($post->ID){ case 页面1_ID: //在这里添加页面1的模板样式 echo '这是页面1'; break; case 页面2_ID: //在这里添加页面2的模板样式 echo '这是页面2'; break; default: //在这里输出默认的page模板 echo '这里是默认的页面模板'; } ?> [/php]
但是如果我们的页面比较多,各个页面的样式不同而且比较复杂的话,使用这种方法就不太合适,管理起来就不太方便。 选择页面模板 通过选择页面模板,我们可以为每个页面都单独指定一个模板文件。方法如下: 一、新建页面模板文件:在主题文件夹下新建一个php文件,文件名任意,比如testpage.php 二、在新建的模板文件中添加模板名,格式如下:
view plaincopy to clipboardprint?
[php] <?php /* Template Name: page test */ ?> [/php]
“Template Name:”后面的字符就是模板名,你可以任意定义。这样wordpress才能知道该文件是一个模板文件,并且模板名称为page test。 三、在后台管理面板新建一个页面或者编辑以前的页面,在属性的模板中选择刚才新建的模板。 这种方法是最常用的一种方法,当然2.9版本还增加了一种方法,就是通过特定的文件名(page-slug.php和page-id.php)来自动选择模板,请参看wordpress2.9模板果然支持page-slug和page-id了 一文。 下面我将通过一些具体的实例进行说明自定义页面的使用方法 自定义关于页面 “关 于”页面是wordpress默认的一个页面(page_id为2),我将使用条件判断的方法对其进行自定义,添加某个分类下的文章列表(比如我的 Domety史记分类的文章列表)。在page.php文件加入一个if($post->ID == 3)和条件判断,代码如下
view plaincopy to clipboardprint?
[php] <?php if($post->ID == 2) : query_posts('cat=3'); if(have_posts()): ?> <li> <h2>Domety史记</h2> <ul> <?php while(have_posts()) : the_post(); ?> <li><a href="<?php the_permalink(); ?>"><?php the_title(); ?></a>(<?php the_time('F j,Y'); ?>)</li> <?php endwhile; ?> </ul> </li> <?php endif;wp_reset_query();endif; ?> [/php]
我的超级链接页面 这 个页面我使用的是模板的方法,首先新建了一个page_superlinks.php模板文件,模板名为”page super links”,然后在后 台新建一个“超级链接”页面,模板选择”page super links”。你们可以看到我的这个页面中没有侧边栏了,是因为在这个模板文件没有调用 get_sidebar函数。代码片段如下
view plaincopy to clipboardprint?
[php] <?php /* Template Name: page super links */ get_header(); ?> [/php] 这里是实现超级链接的代码 [php] <?php get_footer(); ?> [/php]
文档归档页面 该页面显示文章的存档,列出所有月份,就相当于把小工具中的“文章索引模板”单独显示到一个页面中,效果如下 使用的方法仍然是上面的模板法,主题文件夹下新建一个页面模板page-archives.php,文件内容如下
view plaincopy to clipboardprint?
[php] <?php /* Template Name: page archives */ get_header(); echo '<h2>文章索引</h2><ul>'; wp_get_archives('type=monthly&show_post_count=true'); echo '</ul>'; get_footer(); ?> [/php]
在后台添加一个页面,标题可以为“存档”,模板选择”page archives”。 页面跳转 在论坛里看到好多人在询问实现页面跳转的方法,其实我感觉如果只是一个简单的页面跳转的话,最简单的方法就是在导航菜单里直接加入一个链接就行了。下面给出的方法实现的功能是在页面的内容里输入一个网址,然后页面自动跳转到这个网址。 一、主题文件夹下新建一个page-redirect.php文件,文件内容如下
view plaincopy to clipboardprint?
[php] <?php /* Template Name:page redirect */ header('Location:'.$post->post_content); ?> [/php]
二、在后台控制面板新建一个页面,标题根据你的需要填写,比如“论坛”。内容输入需要跳转的网址,比如http://domety.com,模板选择”page redirect”。然后发布即可。

Yii 学习 (转)

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.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事件,日志就是在这个事件中记录的 activeRecord几个占位方法重写的注意点 必须带boolean返回值 如何发布一个资源文件并引用 $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); 或者 <?php $this->beginWidget('CHtmlPurifier'); ?> ...display user-entered content here... <?php $this->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,则在当前位置不显示内容 如果在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的自动完成功能 <?php $this->widget('CAutoComplete', array( 'name'=>'xxx', 'url'=>array('suggestTags'), 'multiple'=>false, 'htmlOptions'=>array('size'=>50), )); ?> 然后在url指定的地址中的方法中如下输出,即可 echo "a\nb\nc" //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显示 设置好 <?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</a></span>', 'separator'=>'>>>' )); ?> 其中breadcrumbs中Controller中的一个属性,如果要出现导航,就要在view中给此属性附值 生成的html如下 <div> <span><a href="http://abc.com">shouye</a> </span>&gt;&gt;&gt;<span>Managde Posts</span>&gt;&gt;&gt; <span>b</span>&gt;&gt;&gt;<span>c</span></div> 所以如果网站用到导航的时候,美工最好把导航代码定义如上 //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'),

yii中常用路径(转)

调用YII框架中jquery:Yii::app()->clientScript->registerCoreScript(‘jquery’); framework/web/js/source的js,其中registerCoreScriptkey调用的文件在framework/web/js/packages.php列表中可以查看   在view中得到当前controller的ID方法:Yii::app()->getController()->id;   在view中得到当前action的ID方法:Yii::app()->getController()->getAction()->id;   yii获取ip地址:Yii::app()->request->userHostAddress;   yii判断提交方式:Yii::app()->request->isPostRequest   得到当前域名: Yii::app()->request->hostInfo   得到proteced目录的物理路径:YII::app()->basePath;   获得上一页的url以返回:Yii::app()->request->urlReferrer;   得到当前url :Yii::app()->request->url;   得到当前home url :Yii::app()->homeUrl   得到当前return url :Yii::app()->user->returnUrl   项目路径:dirname(Yii::app()->BasePath)     一:Yii framework 已经定义的命名空间常量 system: 指向Yii框架目录;  Yii\framework zii: 指向zii library 目录;  Yii\framework\zii application : 指向应用程序基本目录;  protected\ webroot: 指向包含里入口脚本 文件的目录;  .\ ext : 指向包含所有第三方扩展的目录;  \protected\extensions   用法:Yii::getPathOfAlias(‘webroot’) 二:
{full URL}:取得当前的完整路径 Yii::getFrameworkPath()  :YII framework路径
  三: 插入meta信息
复制代码
Yii::app()->clientScript->registerMetaTag(‘keywords’,'关键 字’);   Yii::app()->clientScript->registerMetaTag(‘description’,'一些 描述’);   Yii::app()->clientScript->registerMetaTag(‘author’,'作者’); 示例: <link rel=”alternate” type=”application/rss+xml” href=”http://www.dreamdu.com/feed/” /> 表示为: Yii::app()->clientScript->registerLinkTag(‘alternate’,'application/rss+xml’,$this->createUrl(‘/feed’));
复制代码
  在控制器添加CSS文件或JavaScript文件
Yii::app()->clientScript->registerCssFile(Yii::app()->baseUrl.’/css/my.css’);   Yii::app()->clientScript->registerScriptFile(Yii::app()->baseUrl.’/css/my.js’);
  在view中得到当前controller的ID方法
Yii::app()->getController()->id;
    在view中得到当前action的ID方法:
Yii::app()->getController()->getAction()->id;
  Yii获取ip地址
Yii::app()->request->userHostAddress;
  Yii判断提交方式
Yii::app()->request->isPostRequest
  得到当前域名:
Yii::app()->request->hostInfo
    得到proteced目录的物理路径
Yii::app()->basePath;
    获得上一页的url以返回
Yii::app()->request->urlReferrer;
    得到当前url  
Yii::app()->request->url;
    得到当前home url  
Yii::app()->homeUrl
  得到当前return url  
Yii::app()->user->returnUrl
  项目路径  
dirname(Yii::app()->BasePath)

YII框架总结 (转)

1. Controller向View里面传送数据,View使用Controller的数据 Controller对象的数据,View可以通过$this->varname 来访问。 举例:array("varname1'=>$var1, 'varname2'=>$var2); View中就使用$varname1, $varname2; Controller对象从Model取出来的数据,作为临时变量的话,View通过普通的本地变量使用,$varname。 还有一种特例,Widget对象初始化,Controller传进去的特殊临时变量,Widget有他自己特殊的用法,例如: <?php $this->widget('zii.widgets.CListView', array( 'dataProvider'=>$dataProvider, 'itemView'=>'_bookView', )); ?> 在View中,$data就是$dataProvider提供的数据. 1. 数据访问 两种方式,1是最基本的增删差改(基于单个表的)CActiveRecord类;2是Yii DAO类,包括 CDbConnection : 代表一个数据库连接。 CDbCommand : 代表一条通过数据库执行的 SQL 语句。 CDbDataReader : 代表一个只向前移动的,来自一个查询结果集中的行的流。 CDbTransaction : 代表一个数据库事务。 其中,有简单的AR封装,$dataProvider=new CActiveDataProvider('Book'); 以下是参考内容: 1. Yii Framework] 如何获取当前controller的名称? 下面语句就可以获取当前控制器的名称了! Yii::app()->controller->id 2. yii 如何使用第三方插件 第一,比如说,我们要使用 Zend framework的东西。我们把zend framework解压到 prtected/vendors里面,现在的文件夹为 protected/vendors/Zend/Search/Lucene.php 第二,在controller文件的头部,插入下面代码。 Yii::import(’application.vendors.*’); require once(’Zend/Search/Lucene.php’); 上面代码包含了Lucene.php这个类文件。因为我们用到的是相对路径,所以我们需要改变PHP加载文件的路径,Yii::import 一定要在require_once 之前。 第三,一旦我们设置好了,我们就可以在controller里面使用了。比如说 $lucene=new Zend Search Lucene($pathOfIndex); $hits=$lucene->find(strtolower($keyword)); 3. yii中如何在查询的时候使用数据库函数 比如要使用mySQL中的md5函数, Test::model()->findAll(new CDbExpression("md5(name) =1")); 4. yii的controller中外挂action 创建 class UpdateAction extends CAction { public function run() { // place the action logic here } } 调用 class PostController extends CController { public function actions() { return array( ’edit’=>’application.controllers.post.UpdateAction’, ); } } 5. Yii创建widget class MyWidget extends CWidget { public function init() { // this method is called by CController::beginWidget() } public function run() { // this method is called by CController::endWidget() } } 通常,widget的视图是是放在components/views里面的,通过CWidget::render()来传递参数的 6. CWidget::init()与CWidget::run()的区别 要创建一个新的挂件(widget),我们主要是要继承两个方法:CWidget::init()和 CWidget::run(), CWidget::init 调用是发生在我们使用 $this->beginWidget 将挂件插入到一个view里面, CWidget::run 调用是发生在我们使用 $this->endWidget 这个方法的时候。 如果我们想捕捉和处理两者之间的方法核查办上显示的内容,我们可以在CWidget::init开始输出缓冲,然后在CWidget::run中检索缓冲输出 并作进一步处理。 7. Yii如何使用theme 在main.php 里面配置 return array( ’theme’=>’basic’, ...... ); 要使用theme里面的资源的话,比如说images, js, css, 应该这样, Yii::app()->theme->baseUrl.”/images/FileName.gif” Yii::app()->Theme->baseUrl.”/css/default/common.css” 8.Yii 如何在当前页面注册css和js文件 $cs=Yii::app()->clientScript; $cs->registerCssFile($cssFile); $cs->registerScriptFile($jsFile); 9.Yii Captcha验证码的使用方法 假设使用的model名字为Comment Model里面 Public function rules() { Return array( ...... Array('verifyCode', 'captcha', 'on'=>'insert', 'allowEmpty'=>!Yii::app()->user->isGuest || !extension_loaded('gd')), ); } View里面 <form action=”/test/xyz” method=”post”> <input type=”text” name=”comment[verifyCode]”/> </form> Controller里面 Public function xyz() { $comment = new Comment; $comment->validate('insert');//因为是insert的时候才会用到captcha,所以要加上参数'insert' } 10. 如何调用extension扩展 Components的方法 引入以及定义: 在config.php文件里面 ’components’=>array( ’xyz’=>array( ’class’=>’ext.xyz.XyzClass’, ’property1’=>’value1’, ’property2’=>’value2’, ), // other component configurations ), 使用方法: 在 任何地方,使用Yii::app()->xyz,就可以直接使用xyz这个component了,而component的加载方式是 lazilycreated的,只要我们不是在preload=array()里面定义,那么就是,当第一次使用的时候,才会实例化的,所以不用担心说把 它放在config.php里面会影响性能。 11. Yii 数据保存时自动插入createTime和updateTime Yii 1.1 version之后,可以直接这样: public function behaviors(){ return array( 'CTimestampBehavior' => array( 'class' => 'zii.behaviors.CTimestampBehavior', 'createAttribute' => 'create_time_attribute', 'updateAttribute' => 'update_time_attribute', ) ); } 如果model里面已经在使用public function behaviors(),记得要在前面加上parent::behaviors($on); 12. Yii 数据库查询找出最新5个发布的内容 在数据查询的时候,出现下面的是什么意思? $posts=Post::model()->published()->recently()->findAll(); 这个是叫做named scope, 每个命名范围被声明为一个可以被用来初始化CDbCriteria阵列实例。 如要下面的例子 class Post extends CActiveRecord { ...... public function scopes() { return array( ’published’=>array( ’condition’=>’status=1’, ), ’recently’=>array( ’order’=>’createTime DESC’, ’limit’=>5, ), ); } } 而$posts=Post::model()->published()->recently()->findAll();的意思就是找出最新的status为1的post的5条记录 13. 在views里面如何调用本controller的方法,获取一定的值 直接在views里面使用$this->method(),如 controller里面: Class PostController extends Ccontroller { Puvlic function actionList(){....} Public function getTitle(){return 'test title';} } views的list.php <?php echo $this->getTitle();?> 这样就可以调用本controller的方法了 14. Yii framework已经定义的命名空间常量 system: Yii framework directory application: application's base directory webroot: the directory containing the entry script file ext: directory of extensions system: 指向 Yii 框架目录; zii: 指向 zii library 目录; application: 指向应用程序 基本目录(base directory); webroot: 指向包含里 入口脚本 文件的目录. 此别名自 1.0.3 版起生效. ext: 指向包含所有第三方扩展的目录, 从版本 1.0.8 可用; 15. yii中如何不加载layout 可以使用renderPartial()来代替render() 16. yii中向widget传值 $this->widget('CMaskedTextField',array('mask'=>'99/99/9999'));

Yii 学习 CDbCriteria常用方法(转)

Yii的Active Recorder包装了很多。 特别是把SQL中 把where,order,limit,IN/not IN,like等常用短句都包含进CDbCriteria这个类中去,这样整个代码会比较规范,一目了然。 $criteria =newCDbCriteria; $criteria->addCondition(“id=1″); //查询条件,即where id =1 $criteria->addInCondition(‘id’, array(1,2,3,4,5));//代表where id IN (1,23,,4,5,); $criteria->addNotInCondition(‘id’,array(1,2,3,4,5));//与上面正好相法,是NOT IN $criteria->addCondition(‘id=1′,’OR’);//这是OR条件,多个条件的时候,该条件是OR而非AND $criteria->addSearchCondition(‘name’,'分类’);//搜索条件,其实代表了。。where name like ‘%分类%’ $criteria->addBetweenCondition(‘id’, 1, 4);//between1 and 4 $criteria->compare(‘id’,1);   //这个方法比较特殊,他会根据你的参数自动处理成addCondition或者addInCondition, //即如果第二个参数是数组就会调用addInCondition $criteria->addCondition(“id = :id”); $criteria->params[':id']=1; $criteria->select = ‘id,parentid,name’;//代表了要查询的字段,默认select=’*'; $criteria->join = ‘xxx’; //连接表 $criteria->with = ‘xxx’;//调用relations $criteria->limit =10;   //取1条数据,如果小于0,则不作处理 $criteria->offset =1;   //两条合并起来,则表示 limit 10 offset1,或者代表了。limit 1,10 $criteria->order = ‘xxx DESC,XXX ASC’ ;//排序条件 $criteria->group = ‘group 条件’; $criteria->having = ‘having 条件 ‘; $criteria->distinct = FALSE;//是否唯一查询

在CI中使用自定义钩子

CI提供的钩子的功能,但是在手册中并没有做为类库对外提供。其实是可以直接使用的。1、config.php文件中启用hook [php] $config['enable_hooks'] = TRUE; [/php] 2、配置hooks.php配置文件 [php] $hook['after_login'] = <a href="http://www.php.net/array">array</a>(                                 'class'    => 'MyClass',                                 'function' => 'Myfunction',                                 'filename' => 'Myclass.php',                                 'filepath' => 'hooks'                                 ); [/php] 3、在hooks目录下新建MyClass.php,内容如下: [php] <?php class MyClass{     function Myfunction()     {         echo "TEST";     } } ?> [/php] 4、使用 在任意控制器里: [php] $this->hooks->_call_hook('after_login'); [/php] 访问这个控制器,页面上打出TEST,说明钩子被调用。 注意到这里的挂载点,after_login是我们自己随便定义的。我们可以在用户登录后,加入这样一句代码,方便将来处理登录后的一些操作。 转自CI中文论坛

Yii 学习 项目目录结构

testdrive/
   index.php                 Web 应用入口脚本文件
   index-test.php            功能测试使用的入口脚本文件
   assets/                   包含公开的资源文件
   css/                      包含 CSS 文件
   images/                   包含图片文件
   themes/                   包含应用主题
   protected/                包含受保护的应用文件
      yiic                   yiic 命令行脚本
      yiic.bat               Windows 下的 yiic 命令行脚本
      yiic.php               yiic 命令行 PHP 脚本
      commands/              包含自定义的 'yiic' 命令
         shell/              包含自定义的 'yiic shell' 命令
      components/            包含可重用的用户组件
         Controller.php      所有控制器类的基础类
         Identity.php        用来认证的 'Identity' 类
      config/                包含配置文件
         console.php         控制台应用配置
         main.php            Web 应用配置
         test.php            功能测试使用的配置
      controllers/           包含控制器的类文件
         SiteController.php  默认控制器的类文件
      data/                  包含示例数据库
         schema.mysql.sql    示例 MySQL 数据库
         schema.sqlite.sql   示例 SQLite 数据库
         testdrive.db        示例 SQLite 数据库文件
      extensions/            包含第三方扩展
      messages/              包含翻译过的消息
      models/                包含模型的类文件
         LoginForm.php       'login' 动作的表单模型
         ContactForm.php     'contact' 动作的表单模型
      runtime/               包含临时生成的文件
      tests/                 包含测试脚本
      views/                 包含控制器的视图和布局文件
         layouts/            包含布局视图文件
            main.php         所有视图的默认布局
            column1.php      使用单列页面使用的布局
            column2.php      使用双列的页面使用的布局
         site/               包含 'site' 控制器的视图文件
            pages/           包含 "静态" 页面
               about.php     "about" 页面的视图
            contact.php      'contact' 动作的视图
            error.php        'error' 动作的视图(显示外部错误)
            index.php        'index' 动作的视图
            login.php        'login' 动作的视图
         system/             包含系统视图文件