ThinkPHP控制器学习(一)

WBOY
Freigeben: 2016-06-20 12:27:16
Original
943 Leute haben es durchsucht

星空

多层控制器的使用

TP3.23对控制器做了更加细致的分层,除了默认的Controller层,还可以自定义事件控制层Event。

建立的方法是:

  • 在模块第一级目录,即Controller层的同级目录,新建文件夹Event
  • 在Event文件夹中,新建文件 AdminEvent.class.php
  • 在新建文件中输入代码
namespace Admin\Event;// use Think\Controller;  // 如果不使用到controller的功能 // 可以不用添加该命名空间引用 也不用继承controllerclass AdminEvent extents Controller {    public function test() {        echo 'Hello World !';    }}
Nach dem Login kopieren

系统默认的控制器(即访问控制器)是Controller,因此在浏览器输入http://localhost/forum/index.php/Admin/admin/test是无法进行访问的,Event控制器仅在内部方法方法中进行访问。当然可以通过DEFAULT_C_LAYER来修改访问控制器的名称。但建议不这么做。

定义了事件控制器,对系统模块的构建有很大的好处。可以将浏览器的请求与内部事件处理相隔离,使整个业务逻辑看上去更为简洁明了。

如一个后台系统,有多个页面,都需要对用户是否登录进行检测,如果这个检测都在Controller进行的话,整个业务逻辑会显得很混乱,而且可能会有代码冗余。这时可以通过Event事件控制器来定义一个检测方法,然后在Controller层进行调用,这样代码不仅更为简练,业务逻辑也更为清晰。

Event控制器

namespace Admin\Event;use Think\Controller;      class AdminEvent extents Controller {    /**     *  登录检测  检测session中用户     *  @param    类型    参数名 含义     *  @return 类型       */    function checkLogin() {        $value = session('UserName');        if(!isset($value)) {            $this->error('请登录后访问' , U('admin/login'));        }    }}
Nach dem Login kopieren

Controller控制器

namespace Admin\Controller;use Think\Controller;class AdminController extends Controller {    /**      *  后台首页面     */    public function index(){        A('Admin' , 'Event')->checkLogin();        $this->assign('title','后台');           $this->display();       }    /**     -  文章显示控制     -  @param       -  @return      */    public function article() {        A('Admin' , 'Event')->checkLogin();        $this->assign('title','文章-后台');                     $this->display();    }}
Nach dem Login kopieren

这样无论是从那种链接登入后台,都会进行用户检测,不仅提高了安全性,而且代码逻辑清晰。

我们可以将数据库添加操作在Event控制器中完成,在Controller控制器只进行模板输出,接受表单提交。以上。

初始化操作

在Think\Controller.class.php的构造方法中进行视图类实例化后,会检测是否存在一个_initialize方法。如果存在,就会事先调用该方法。

这就是控制器的初始化操作。在控制器类中定义_initialize方法,在操作其他方法之前都会先执行该方法。

如:

namespace Admin\Controller;use Think\Controller;class AdminController extends Controller {    # 初始化操作    public function _initialize() {        echo '页面初始化,请等待<br/>';    }    # 主页    public function index() {        echo 'index';    }}
Nach dem Login kopieren

在浏览器输入http://localhost/forum/index.php/Admin,则会输出

页面初始化,请等待index

有了这个方法,我们上面所讲的登录检测就能够再次进行简化了。即在_initialize方法中进行是否登录的检测。但在_initialize添加了登录检测后,我们不能够将与登录有关的内容放入AdminController控制器中了,否则会不停的跳转到error页面,而无法显示登陆页面。

我们可以将登录与注册同时拿出,新建一个控制器,名字叫UserController。在这个页面进行登陆与注册功能,而error页面跳转到该页面。

UserController

namespace Admin\Controller;use Think\Controller;class UserController extends Controller {    /**     *  用户登录     */     public function login() {        echo '用户登录';     }    /**     *  用户注册     */    public function register() {        echo '用户注册';    }}
Nach dem Login kopieren

AdminController

namespace Admin\Controller;use Think\Controller;class AdminController extends Controller {    # 初始化操作    public function _initialize() {        A('Admin' , 'Event')->checkLogin();    }    /**      *  后台首页面     */    public function index(){                   $this->assign('title','后台');           $this->display();       }    /**     -  文章显示控制     -  @param       -  @return      */    public function article() {                    $this->assign('title','文章-后台');                     $this->display();    }}
Nach dem Login kopieren

AdminEvent

namespace Admin\Event;use Think\Controller;      class AdminEvent extents Controller {    /**     *  登录检测  检测session中用户     *  @param    类型    参数名 含义     *  @return 类型       */    function checkLogin() {        $value = session('UserName');        if(!isset($value)) {            $this->error('请登录后访问' , U('user/login'));        }    }}
Nach dem Login kopieren

这样一个比较完善的后台登陆就完成了(当然还未实现具体的业务逻辑),比较最原始的方法,代码简洁清晰了很多,也完全符合MVC设计思想。

前置与后置

与初始化设置类似,TP提供了前置与后置操作,要进行前后置操作需要真实的方法,系统在执行该方法前会进行检测,如果定义了前后置操作,则会安照顺序进行。定义的方法是在执行方法前加入_before_或_after_,如

namespace Admin\Controller;use Think\Controller;class AdminController extends Controller {    public function _before_index() {}    public function index() {}    public function _after_index() {}}
Nach dem Login kopieren

在访问http://localhost/forum/index.php/Admin的时候,就会先执行_before_index,在执行index,最后执行_after_index。注意的是如果在index中定义了断点方法,如die,exit,跳转方法error,success就不会继续执行_after_index。

这个前后置方法,我暂时还未想到有什么比较有用的地方,先写出来,以后想到了,在进行添加。

空操作与空控制器

按照ThinkPHP的URL命名规则(pathinfo模式),入口文件之后的操作成为

模块控制器操作

如http://servername/index.php/模块/控制器/操作/[参数名/参数值…]

空操作就是指系统找不到url指定的操作方法,此时就会报错,可以使用定义空操作方法来进行避免。如:

namespace Admin\Controller;use Think\Controller;class AdminController extends Controller {    /**      *  后台首页面     */    public function index(){                   $this->assign('title','后台');           $this->display();       }    /**     -  空操作防止     -  @param    无     -  @return   无        */    public function _empty() {        $this->display('404');    }}
Nach dem Login kopieren

定义后,在输入错误的操作方法时,就会显示_empty中的内容。你可以定义一个404页面,在_empty中进行跳转。

空控制器

与空操作类似,访问不存在的控制器就会产生空控制器操作,避免该错误的方法是,在Controller中定义一个EmptyController控制器,在该控制器中同样定义_empty方法。

namespace Admin\Controller;use Think\Controller;class EmptyController extends Controller {    /**     -    空操作防止     -    @param    无     -    @return   无        */    public function _empty() {         $this->display('admin/404');    }}
Nach dem Login kopieren

404页面对于404页面的定制,如果没有特殊需求的同学,建议使用腾讯的404公益页面,只要嵌入一小段JS代码,就会跳转到腾讯的寻找失踪儿童的404页面。虽然可能和你的网站风格不统一,但是每多一个看见该页面的人,就可能加大找回一个失踪儿童的几率。一小段代码,献出一段爱心,你值得拥有。

// 腾讯404页面<script type="text/javascript" src="http://www.qq.com/404/search_children.js" charset="utf-8" homePageUrl="http://yoursite.com/yourPage.html" homePageName="回到我的主页"></script>
Nach dem Login kopieren

20160529162719.png

Action参数绑定

在说Action参数绑定前,先说说与之相关的问题。在学习Action参数绑定之前,我获取GET参数的方式是怎么样?自然是利用$_GET来获取URL上的参数,如:

# 我要点击一个文章链接,来进行文章编辑 url如下http://localhost/forum/index.php/Admin/admin/article/act/edit/aid/2# 然后通过$_GET来获取act aid得到参数,来判断进行的操作以及操作的文章idif($_GET['act'] == 'edit') {    # 文章编辑  利用 aid从数据库取出文章进行编辑}
Nach dem Login kopieren

这种方式虽然可行,却不够优雅,作为一个立志成为攻城狮的程序员来说,要想尽方法使你的代码更加简洁,更加优雅。

这时来看看Action参数绑定,什么是Action参数绑定?

Action参数绑定是通过直接绑定URL地址中的变量作为操作方法的参数, 可以简化方法的定义甚至路由的解析。

说的还挺绕,其实就是把GET形式传递的参数直接绑定到你的操作方法上,你就能够直接访问,而不用通过$_GET。举个例子,还是上面的链接http://localhost/forum/index.php/Admin/admin/article/act/edit/aid/2。定义操作方法如下:

namespace Admin\Controller;use Think\Controller;class AdminController extends Controller {    /**      *  文章显示控制器     *  @param:$act   string   进行的文章操作  默认值空 -> 显示所有文章     *  @param:$aid   int      文章的主键id    默认值0  -> 显示所有文章     */    public function article($act='' , $aid = 0) {        if(empty($act) || empty($aid) ) {            # 显示所有文章        } else {            # 进行想要的操作            echo '你要进行的操作是'.$act.',你想要'.$act.'的文章id是'.$aid;        }    }}
Nach dem Login kopieren

访问上面的链接,结果为

你要进行的操作是edit,你想要edit的文章id是2

这就是Action参数绑定。要使用这种方式需要开启URL_PARAMS_BIND设置(默认设置true)。

Action参数绑定有两种形式

按照变量名绑定 按照变量顺序绑定

将URL_PARAMS_BIND_TYPE的值设置成0,按照变量名绑定,设置成1,按照变量顺序绑定。

按照字面意思也可以理解,按照变量名绑定即寻找get参数时,按照操作方法中定义的变量名去寻找相应的值。如果没有就报错。这也是最常用的方式。

按照变量顺序绑定,即按照url上get参数的顺序去给操作方法上的变量赋值,这样在url上的参数就能够随意变换位置,同时url上get参数也可以隐藏变量名。

举例说明一下:

按照变量名绑定链接 http://localhost/forum/index.php/Admin/admin/article/act/edit/aid/2 那么打印出来的结果

你要进行的操作是edit,你想要edit的文章id是2

链接 http://localhost/forum/index.php/Admin/admin/article/aid/2/act/edit 那么打印出来的结果

你要进行的操作是edit,你想要edit的文章id是2

可以看出无论怎么变换位置,得到的结果是一样的。

按照变量顺序绑定链接 http://localhost/forum/index.php/Admin/admin/article/edit/2 那么打印出来的结果

你要进行的操作是edit,你想要edit的文章id是2

链接 http://localhost/forum/index.php/Admin/admin/article/2/edit 那么打印出来的结果

你要进行的操作是2,你想要edit的文章id是edit

因此如果采用变量顺序绑定时,一定要确保url上的get参数顺序与操作方法上的参数顺序一致。

值得注意的是按照变量名绑定仅对类似于pathinfo方式的地址有效。如pathinfo模式与兼容模式。

伪静态

伪静态是相对于静态页面来说的,主要是为了更好的SEO效果,并不是真正的静态,而是在URL的结尾添加了类似html,htm等的后缀。在TP中默认是开启伪静态的。可以通过URL_HTML_SUFFIX来设置静态的后缀名,如:

'URL_HTML_SUFFIX'=>'shtml'
Nach dem Login kopieren

访问:

http://localhost/forum/index.php/Admin/admin/article/2/edit.shtml
Nach dem Login kopieren

就是有效的,而访问.html则会报错。也可以设置多个静态后缀,使用**|**来进行分割。

// 多个伪静态后缀设置 用|分割'URL_HTML_SUFFIX' => 'html|shtml|htm'
Nach dem Login kopieren

也可以对某些后缀进行禁止访问,利用URL_DENY_SUFFIX

'URL_HTML_SUFFIX' => 'html'
Nach dem Login kopieren

设置后就不能访问任何以html为后缀的url。可以将URL的模式改成rewrite模式来配合伪静态,否则一个链接上既有.php,也有.html看上去很别扭。

注意:使用伪静态模式必须开启httpd.conf的mod_rewrite.so模块。

rewire模式切换

开启rewrite模式需要配合修改apache的重写内容。

打开httpd.conf文件,搜索mod_rewrite.so,将该模块前面的#删除。然后新建.htaccess,放入如下代码:

<IfModule mod_rewrite.c>    RewriteEngine on    RewriteCond %{REQUEST_FILENAME} !-d    RewriteCond %{REQUEST_FILENAME} !-f    RewriteRule ^(.*)$ index.php/$1 [QSA,PT,L]</IfModule>
Nach dem Login kopieren

保存到项目入口文件同级目录。重启apache即可。

apache的部分重写规则

实现伪静态多数是利用apache的URL重写规则(RewriteRule),我不是很懂RewriteRule,只是从网上找来了一些资料。上述代码中,RewriteEngine表示是否开启重写引擎,RewriteCond重写应用条件,RewriteRule表示重写规则。从代码中明显看出,利用了正则表达式的功能。

正则表达式含义

表达式 含义 表达式 含义
0-1个字符 $ 段落结束字符
. 1个字符 \ 转移字符
* 0-x个字符 取反
+ 1-x个字符 () 内存限定传值
^ 段落开始字符 [0-9] 所有数字字符
[a-z] 所有小写字母 [A-Z] 所有大写字母
{n} 重复n次 {n,} 重复n次或更多次
{n,m} 重复n到m次

在被匹配的字符串中$N表示与表达式中第一个()进行匹配,上述代码中RewriteRule中的$1就表示与表达式中的第一个()进行匹配,这样所有index.php/都会被其他字符或空字符匹配。

RewriteRule我确实不怎么懂,网上的资料也很乱,就不多写了,英语好的同学去看官方文档吧。

总结

今天关于控制器的学习就写到这里,还有很多想要说的,但没必要都一一赘述。下次会将阅读源码的一些想法写出来。最后想吐槽一下,简书关于PHP的程序员好像很少,每次找很长时间才能看见一篇关于PHP的,醉了醉了。

Verwandte Etiketten:
Quelle:php.cn
Erklärung dieser Website
Der Inhalt dieses Artikels wird freiwillig von Internetnutzern beigesteuert und das Urheberrecht liegt beim ursprünglichen Autor. Diese Website übernimmt keine entsprechende rechtliche Verantwortung. Wenn Sie Inhalte finden, bei denen der Verdacht eines Plagiats oder einer Rechtsverletzung besteht, wenden Sie sich bitte an admin@php.cn
Beliebte Tutorials
Mehr>
Neueste Downloads
Mehr>
Web-Effekte
Quellcode der Website
Website-Materialien
Frontend-Vorlage