多应用+插件架构,代码干净,二开方便,首家独创一键云编译技术,文档视频完善,免费商用码云13.8K 广告
我们已经了解了如何定义路由规则,下面我们来举个例子加深印象。 假设我们定义了News控制器如下(代码实现仅供参考): ~~~ ~~~ namespace Home\Controller; use Think\Controller; class NewsController extends Controller{ public function read(){ $New = M('New'); if(isset($_GET['id'])) { // 根据id查询结果 $data = $New->find($_GET['id']); }elseif(isset($_GET['name'])){ // 根据name查询结果 $data = $New->getByName($_GET['name']); } $this->data = $data; $this->display(); } public function archive(){ $New = M('New'); $year = $_GET['year']; $month = $_GET['month']; $begin_time = strtotime($year . $month . "01"); $end_time = strtotime("+1 month", $begin_time); $map['create_time'] = array(array('gt',$begin_time),array('lt',$end_time)); $map['status'] = 1; $list = $New->where($map)->select(); $this->list = $list; $this->display(); } } ~~~ ~~~ 定义路由规则如下: ~~~ ~~~ 'URL_ROUTER_ON' => true, //开启路由 'URL_ROUTE_RULES' => array( //定义路由规则 'new/:id\d' => 'News/read', 'new/:name' => 'News/read', 'new/:year\d/:month\d' => 'News/archive', ), ~~~ ~~~ 然后,我们访问: `http://serverName/index.php/Home/new/8` 会匹配到第一个路由规则,实际执行的效果等效于访问: `http://serverName/index.php/Home/News/read/id/8` 当访问: `http://serverName/index.php/Home/new/hello` 会匹配到第二个路由规则,实际执行的效果等效于访问: `http://serverName/index.php/Home/News/read/name/hello` 那么如果访问: `http://serverName/index.php/Home/new/2012/03` 是否会匹配第三个路由规则呢?我们期望的实际执行的效果能够等效于访问: `http://serverName/index.php/Home/News/archive/year/2012/month/03` 事实上却没有,因为`http://serverName/index.php/Home/new/2012/`这个URL在进行路由匹配过程中已经优先匹配到了第一个路由规则了,把2012当成id的值传入了,这种情况属于路由规则的冲突,解决办法有两个: **1、调整定义顺序** 路由定义改成: ~~~ ~~~ 'URL_ROUTE_RULES' => array( //定义路由规则 'new/:year\d/:month\d' => 'News/archive', 'new/:id\d' => 'News/read', 'new/:name' => 'News/read', ), ~~~ ~~~ 接下来,当我们再次访问: `http://serverName/index.php/Home/new/2012/03` 的时候,达到了预期的访问效果。所以如果存在可能规则冲突的情况,尽量把规则复杂的规则定义放到前面,确保最复杂的规则可以优先匹配到。但是如果路由规则定义多了之后,仍然很容易混淆,所以需要寻找更好的解决办法。 **2、利用完全匹配功能** 现在我们来利用路由的完全匹配定义功能,把路由定义改成: ~~~ ~~~ 'URL_ROUTE_RULES' => array( //定义路由规则 'new/:id\d$' => 'News/read', 'new/:name$' => 'News/read', 'new/:year\d/:month\d$' => 'News/archive', ), ~~~ ~~~ 在规则最后加上$符号之后,表示完整匹配当前的路由规则,就可以避免规则定义的冲突了。对于规则路由来说,简单的理解就是URL里面的参数数量或者类型约束要完全一致。 所以,如果我们访问 `http://serverName/index.php/Home/new/2012/03/01` 的话,是不会匹配成功任何一条路由的。 **3、利用正则路由** 当然,解决问题的办法总是不止一种,对于复杂的情况,我们不要忘了使用正则路由规则定义,在你找不到解决方案的时候,正则路由总能帮到你。 要实现上面的同样路由功能的话,还可以用下面的规则定义: ~~~ ~~~ 'URL_ROUTE_RULES' => array( //定义路由规则 '/^new\/(\d+)$/' => 'News/read?id=:1', '/^new\/(\w+)$/' => 'News/read?name=:1', '/^new\/(\d{4})\/(\d{2})$/' => 'News/achive?year=:1&month=:2', ), ~~~ ~~~