> 백엔드 개발 > PHP 튜토리얼 > ThinkPHP控制器学习(一)

ThinkPHP控制器学习(一)

WBOYWBOYWBOYWBOYWBOYWBOYWBOYWBOYWBOYWBOYWBOYWBOYWB
풀어 주다: 2016-06-20 12:27:16
원래의
979명이 탐색했습니다.

星空

多层控制器的使用

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 !';    }}
로그인 후 복사

系统默认的控制器(即访问控制器)是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'));        }    }}
로그인 후 복사

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();    }}
로그인 후 복사

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

我们可以将数据库添加操作在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';    }}
로그인 후 복사

在浏览器输入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 '用户注册';    }}
로그인 후 복사

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();    }}
로그인 후 복사

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'));        }    }}
로그인 후 복사

这样一个比较完善的后台登陆就完成了(当然还未实现具体的业务逻辑),比较最原始的方法,代码简洁清晰了很多,也完全符合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() {}}
로그인 후 복사

在访问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');    }}
로그인 후 복사

定义后,在输入错误的操作方法时,就会显示_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');    }}
로그인 후 복사

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>
로그인 후 복사

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从数据库取出文章进行编辑}
로그인 후 복사

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

这时来看看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;        }    }}
로그인 후 복사

访问上面的链接,结果为

你要进行的操作是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'
로그인 후 복사

访问:

http://localhost/forum/index.php/Admin/admin/article/2/edit.shtml
로그인 후 복사

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

// 多个伪静态后缀设置 用|分割'URL_HTML_SUFFIX' => 'html|shtml|htm'
로그인 후 복사

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

'URL_HTML_SUFFIX' => 'html'
로그인 후 복사

设置后就不能访问任何以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>
로그인 후 복사

保存到项目入口文件同级目录。重启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的,醉了醉了。

관련 라벨:
원천:php.cn
본 웹사이트의 성명
본 글의 내용은 네티즌들의 자발적인 기여로 작성되었으며, 저작권은 원저작자에게 있습니다. 본 사이트는 이에 상응하는 법적 책임을 지지 않습니다. 표절이나 침해가 의심되는 콘텐츠를 발견한 경우 admin@php.cn으로 문의하세요.
인기 튜토리얼
더>
최신 다운로드
더>
웹 효과
웹사이트 소스 코드
웹사이트 자료
프론트엔드 템플릿