溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊(cè)×
其他方式登錄
點(diǎn)擊 登錄注冊(cè) 即表示同意《億速云用戶服務(wù)條款》

怎么在thinkphp5中實(shí)現(xiàn)一個(gè)路由功能

發(fā)布時(shí)間:2021-01-05 15:08:24 來源:億速云 閱讀:231 作者:Leah 欄目:開發(fā)技術(shù)

這篇文章將為大家詳細(xì)講解有關(guān)怎么在thinkphp5中實(shí)現(xiàn)一個(gè)路由功能,文章內(nèi)容質(zhì)量較高,因此小編分享給大家做個(gè)參考,希望大家閱讀完這篇文章后對(duì)相關(guān)知識(shí)有一定的了解。

URL訪問

ThinkPHP采用單一入口模式訪問應(yīng)用,對(duì)應(yīng)用的所有請(qǐng)求都定向到應(yīng)用的入口文件,系統(tǒng)會(huì)從URL參數(shù)中解析當(dāng)前請(qǐng)求的模塊、控制器和操作,下面是一個(gè)標(biāo)準(zhǔn)的URL訪問格式:

http://domainName/index.php/模塊/控制器/操作

其中index.php就稱之為應(yīng)用的入口文件(注意入口文件可以被隱藏,后面會(huì)提到)

模塊在ThinkPHP中的概念其實(shí)就是應(yīng)用目錄下面的子目錄,而官方的規(guī)范是目錄名小寫,因此模塊全部采用小寫命名,無論URL是否開啟大小寫轉(zhuǎn)換,模塊名都會(huì)強(qiáng)制小寫

應(yīng)用的index模塊的Index控制器定義如下:

<?php
namespace app\index\controller;
class Index
{
  public function index()
  {
    return 'index';
  }
  public function hello($name = 'World')
  {
    return 'Hello,' . $name . '!';
  }
}

如果直接訪問入口文件的話,由于URL中沒有模塊、控制器和操作,因此系統(tǒng)會(huì)訪問默認(rèn)模塊(index)下面的默認(rèn)控制器(Index)的默認(rèn)操作(index),因此下面的訪問是等效的:

http://tp5.com/index.php
http://tp5.com/index.php/index/index/index

如果要訪問控制器的hello方法,則需要使用完整的URL地址

http://tp5.com/index.php/index/index/hello/name/thinkphp

訪問URL地址后頁(yè)面輸出結(jié)果為:

Hello,thinkphp!

由于name參數(shù)為可選參數(shù),因此也可以使用

http://tp5.com/index.php/index/index/hello

訪問URL地址后頁(yè)面輸出結(jié)果為:

Hello,World!

默認(rèn)情況下,URL地址中的控制器和操作名是不區(qū)分大小寫的,因此下面的訪問其實(shí)是等效的:

http://tp5.com/index.php/index/Index/Index
http://tp5.com/index.php/index/INDEX/INDEX

如果控制器是駝峰的,例如定義一個(gè)HelloWorld控制器(application/index/controller/HelloWorld.php): 

<?php
namespace app\index\controller;
class HelloWorld
{
  public function index($name = 'World')
  {
    return 'Hello,' . $name . '!';
  }
}

正確的URL訪問地址(該地址可以使用url方法生成)應(yīng)該是

http://tp5.com/index.php/index/hello_world/index

系統(tǒng)會(huì)自動(dòng)定位到HelloWorld控制器類去操作

如果使用

http://tp5.com/index.php/index/HelloWorld/index

將會(huì)報(bào)錯(cuò),并提示Helloworld控制器類不存在

如果希望嚴(yán)格區(qū)分大小寫訪問(這樣就可以支持駝峰法進(jìn)行控制器訪問),可以在應(yīng)用配置文件中設(shè)置:

// 關(guān)閉URL自動(dòng)轉(zhuǎn)換(支持駝峰訪問控制器)
'url_convert' => false,

關(guān)閉URL自動(dòng)轉(zhuǎn)換之后,必須使用下面的URL地址訪問(控制器名稱必須嚴(yán)格使用控制器類的名稱,不包含控制器后綴):

http://tp5.com/index.php/index/Index/index
http://tp5.com/index.php/index/HelloWorld/index

如果服務(wù)器環(huán)境不支持pathinfo方式的URL訪問,可以使用兼容方式,例如:

http://tp5.com/index.php?s=/index/Index/index

其中變量s的名稱的可以配置的

5.0不再支持普通的URL訪問方式,所以下面的訪問是無效的,你會(huì)發(fā)現(xiàn)無論輸入什么,訪問的都是默認(rèn)的控制器和操作

http://tp5.com/index.php?m=index&c=Index&a=hello

參數(shù)傳入

通過操作方法的參數(shù)綁定功能,可以實(shí)現(xiàn)自動(dòng)獲取URL的參數(shù),仍然以上面的控制器為例,控制器代碼如下:

<?php
namespace app\index\controller;
class Index
{
  public function index()
  {
    return 'index';
  }
  public function hello($name = 'World')
  {
    return 'Hello,' . $name . '!';
  }
}

當(dāng)我們?cè)L問

http://tp5.com/index.php/index/index/hello

就是訪問app\index\controller\Index控制器類的hello方法,因?yàn)闆]有傳入任何參數(shù),name參數(shù)就使用默認(rèn)值World。如果傳入name參數(shù),則使用:

http://tp5.com/index.php/index/index/hello/name/thinkphp

頁(yè)面輸出結(jié)果為:

Hello,thinkphp!

現(xiàn)在給hello方法增加第二個(gè)參數(shù):

public function hello($name = 'World', $city = '')
  {
    return 'Hello,' . $name . '! You come from ' . $city . '.';
  }

訪問地址為http://tp5.com/index.php/index/index/hello/name/thinkphp/city/shanghai

頁(yè)面輸出結(jié)果為:

Hello,thinkphp! You come from shanghai.

可以看到,hello方法會(huì)自動(dòng)獲取URL地址中的同名參數(shù)值作為方法的參數(shù)值,而且這個(gè)參數(shù)的傳入順序不受URL參數(shù)順序的影響,例如下面的URL地址輸出的結(jié)果和上面是一樣的:

http://tp5.com/index.php/index/index/hello/city/shanghai/name/thinkphp

或者使用http://tp5.com/index.php/index/index/hello?city=shanghai&name=thinkphp

還可以進(jìn)一步對(duì)URL地址做簡(jiǎn)化,前提就是我們必須明確參數(shù)的順序代表的變量,我們更改下URL參數(shù)的獲取方式,把應(yīng)用配置文件中的url_param_type參數(shù)的值修改如下:

// 按照參數(shù)順序獲取
'url_param_type' => 1,

現(xiàn)在,URL的參數(shù)傳值方式就變成了嚴(yán)格按照操作方法的變量定義順序來傳值了,也就是說我們必須使用下面的URL地址訪問才能正確傳入name和city參數(shù)到hello方法:http://tp5.com/index.php/index/index/hello/thinkphp/shanghai

頁(yè)面輸出結(jié)果為:

Hello,thinkphp! You come from shanghai.

如果改變參數(shù)順序?yàn)閔ttp://tp5.com/index.php/index/index/hello/shanghai/thinkphp

頁(yè)面輸出結(jié)果為:

Hello,shanghai! You come from thinkphp.

顯然不是我們預(yù)期的結(jié)果。

同樣,我們?cè)噲D通過http://tp5.com/index.php/index/index/hello/name/thinkphp/city/shanghai

訪問也不會(huì)得到正確的結(jié)果

[注意]按順序綁定參數(shù)的話,操作方法的參數(shù)只能使用URL pathinfo變量,而不能使用get或者post變量

隱藏入口

可以去掉URL地址里面的入口文件index.php,但是需要額外配置WEB服務(wù)器的重寫規(guī)則。

以Apache為例,需要在入口文件的同級(jí)添加.htaccess文件(官方默認(rèn)自帶了該文件),內(nèi)容如下

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

如果用的phpstudy,規(guī)則如下:

<IfModule mod_rewrite.c> 
Options +FollowSymlinks -Multiviews 
RewriteEngine on 
RewriteCond %{REQUEST_FILENAME} !-d 
RewriteCond %{REQUEST_FILENAME} !-f 
RewriteRule ^(.*)$ index.php [L,E=PATH_INFO:$1] 
</IfModule>

接下來就可以使用下面的URL地址訪問了

http://tp5.com/index/index/index
http://tp5.com/index/index/hello

如果使用的apache版本使用上面的方式無法正常隱藏index.php,可以嘗試使用下面的方式配置.htaccess文件:

<IfModule mod_rewrite.c>
Options +FollowSymlinks -Multiviews
RewriteEngine on
RewriteCond %{REQUEST_FILENAME} !-d
RewriteCond %{REQUEST_FILENAME} !-f
RewriteRule ^(.*)$ index.php?/$1 [QSA,PT,L]
</IfModule>

如果是Nginx環(huán)境的話,可以在Nginx.conf中添加:

location / { // …..省略部分代碼
  if (!-e $request_filename) {
    rewrite ^(.*)$ /index.php?s=/$1 last;
    break;
  }
}

定義路由

URL地址里面的index模塊怎么才能省略呢,默認(rèn)的URL地址顯得有點(diǎn)長(zhǎng),下面就來說說如何通過路由簡(jiǎn)化URL訪問。

我們?cè)诼酚啥x文件(application/route.php)里面添加一些路由規(guī)則,如下:

return [
  // 添加路由規(guī)則 路由到 index控制器的hello操作方法
  'hello/:name' => 'index/index/hello',
];

該路由規(guī)則表示所有hello開頭的并且?guī)?shù)的訪問都會(huì)路由到index控制器的hello操作方法。

路由之前的URL訪問地址為:http://tp5.com/index/index/hello/name/thinkphp

定義路由后就只能訪問下面的URL地址http://tp5.com/hello/thinkphp

[注意]定義路由規(guī)則后,原來的URL地址將會(huì)失效,變成非法請(qǐng)求。

但這里有一個(gè)小問題,如果我們只是訪問http://tp5.com/hello

將發(fā)生錯(cuò)誤

事實(shí)上這是由于路由沒有正確匹配到,我們修改路由規(guī)則如下:

return [
  // 路由參數(shù)name為可選
  'hello/[:name]' => 'index/hello',
];

使用[]把路由規(guī)則中的變量包起來,就表示該變量為可選,接下來就可以正常訪問了http://tp5.com/hello

當(dāng)name參數(shù)沒有傳入值的時(shí)候,hello方法的name參數(shù)有默認(rèn)值World,所以輸出的內(nèi)容為 Hello,World!

除了路由配置文件中定義之外,還可以采用動(dòng)態(tài)定義路由規(guī)則的方式定義,例如在路由配置文件(application/route.php)的開頭直接添加下面的方法:

use think\Route;
Route::rule('hello/:name', 'index/hello');

完成的效果和使用配置方式定義是一樣的。

無論是配置方式還是通過Route類的方法定義路由,都統(tǒng)一放到路由配置文件application/route.php文件中

[注意]路由配置不支持在模塊配置文件中設(shè)置

【完整匹配】

前面定義的路由是只要以hello開頭就能進(jìn)行匹配,如果需要完整匹配,可以使用下面的定義:

return [
  // 路由參數(shù)name為可選
  'hello/[:name]$' => 'index/hello',
];

當(dāng)路由規(guī)則以$結(jié)尾的時(shí)候就表示當(dāng)前路由規(guī)則需要完整匹配。

當(dāng)我們?cè)L問下面的URL地址的時(shí)候:

http://tp5.com/hello // 正確匹配
http://tp5.com/hello/thinkphp // 正確匹配
http://tp5.com/hello/thinkphp/val/value // 不會(huì)匹配

【閉包定義】

還支持通過定義閉包為某些特殊的場(chǎng)景定義路由規(guī)則,例如:

return [
  // 定義閉包
  'hello/[:name]' => function ($name) {
    return 'Hello,' . $name . '!';
  },
];

或者

use think\Route;
Route::rule('hello/:name', function ($name) {
  return 'Hello,' . $name . '!';
});

[注意]閉包函數(shù)的參數(shù)就是路由規(guī)則中定義的變量

因此,當(dāng)訪問下面的URL地址:http://tp5.com/hello/thinkphp

會(huì)輸出

Hello,thinkphp!

【設(shè)置URL分隔符】

如果需要改變URL地址中的pathinfo參數(shù)分隔符,只需要在應(yīng)用配置文件(application/config.php)中設(shè)置:

// 設(shè)置pathinfo分隔符
'pathinfo_depr'     => '-',

路由規(guī)則定義無需做任何改變,我們就可以訪問下面的地址:http://tp5.com/hello-thinkphp

【路由參數(shù)】

還可以約束路由規(guī)則的請(qǐng)求類型或者URL后綴之類的條件,例如:

return [
  // 定義路由的請(qǐng)求類型和后綴
  'hello/[:name]' => ['index/hello', ['method' => 'get', 'ext' => 'html']],
];

上面定義的路由規(guī)則限制了必須是get請(qǐng)求,而且后綴必須是html的,所以下面的訪問地址:

http://tp5.com/hello // 無效
http://tp5.com/hello.html // 有效
http://tp5.com/hello/thinkphp // 無效
http://tp5.com/hello/thinkphp.html // 有效

【變量規(guī)則】

接下來,嘗試一些復(fù)雜的路由規(guī)則定義滿足不同的路由變量。在此之前,首先增加一個(gè)控制器類如下:

<?php
namespace app\index\controller;
class Blog
{
  public function get($id)
  {
    return '查看id=' . $id . '的內(nèi)容';
  }
  public function read($name)
  {
    return '查看name=' . $name . '的內(nèi)容';
  }
  public function archive($year, $month)
  {
    return '查看' . $year . '/' . $month . '的歸檔內(nèi)容';
  }
}

添加如下路由規(guī)則:

return [
  'blog/:year/:month' => ['blog/archive', ['method' => 'get'], ['year' => '\d{4}', 'month' => '\d{2}']],
  'blog/:id'     => ['blog/get', ['method' => 'get'], ['id' => '\d+']],
  'blog/:name'    => ['blog/read', ['method' => 'get'], ['name' => '\w+']],
];

在上面的路由規(guī)則中,我們對(duì)變量進(jìn)行的規(guī)則約束,變量規(guī)則使用正則表達(dá)式進(jìn)行定義。

我們看下幾種URL訪問的情況

// 訪問id為5的內(nèi)容
http://tp5.com/blog/5
// 訪問name為thinkphp的內(nèi)容
http://tp5.com/blog/thinkphp
// 訪問2015年5月的歸檔內(nèi)容
http://tp5.com/blog/2015/05

 【路由分組】

上面的三個(gè)路由規(guī)則由于都是blog打頭,所以我們可以做如下的簡(jiǎn)化:

return [
  '[blog]' => [
    ':year/:month' => ['blog/archive', ['method' => 'get'], ['year' => '\d{4}', 'month' => '\d{2}']],  
    ':id'     => ['blog/get', ['method' => 'get'], ['id' => '\d+']],
    ':name'    => ['blog/read', ['method' => 'get'], ['name' => '\w+']],
  ],
];

對(duì)于這種定義方式,我們稱之為路由分組,路由分組一定程度上可以提高路由檢測(cè)的效率

【復(fù)雜路由】

有時(shí)候,還需要對(duì)URL做一些特殊的定制,例如如果要同時(shí)支持下面的訪問地址

http://tp5.com/blog/thinkphp
http://tp5.com/blog-2015-05

我們只要稍微改變路由定義規(guī)則即可:

return [
  'blog/:id'      => ['blog/get', ['method' => 'get'], ['id' => '\d+']],
  'blog/:name'     => ['blog/read', ['method' => 'get'], ['name' => '\w+']],
  'blog-<year>-<month>' => ['blog/archive', ['method' => 'get'], ['year' => '\d{4}', 'month' => '\d{2}']],
];

對(duì) blog-<year>-<month> 這樣的非正常規(guī)范,我們需要使用<變量名>這樣的變量定義方式,而不是 :變量名方式。

簡(jiǎn)單起見,我們還可以把變量規(guī)則統(tǒng)一定義,例如:

return [
  // 全局變量規(guī)則定義
  '__pattern__'     => [
    'name' => '\w+',
    'id'  => '\d+',
    'year' => '\d{4}',
    'month' => '\d{2}',
  ],
  // 路由規(guī)則定義
  'blog/:id'      => 'blog/get',
  'blog/:name'     => 'blog/read',
  'blog-<year>-<month>' => 'blog/archive',
];

在__pattern__中定義的變量規(guī)則我們稱之為全局變量規(guī)則,在路由規(guī)則里面定義的變量規(guī)則我們稱之為局部變量規(guī)則,如果一個(gè)變量同時(shí)定義了全局規(guī)則和局部規(guī)則的話,當(dāng)前的局部規(guī)則會(huì)覆蓋全局規(guī)則的,例如:

return [
  // 全局變量規(guī)則
  '__pattern__'     => [
    'name' => '\w+',
    'id'  => '\d+',
    'year' => '\d{4}',
    'month' => '\d{2}',
  ],

  'blog/:id'      => 'blog/get',
  // 定義了局部變量規(guī)則
  'blog/:name'     => ['blog/read', ['method' => 'get'], ['name' => '\w{5,}']],
  'blog-<year>-<month>' => 'blog/archive',
];

URL生成

定義路由規(guī)則之后,可以通過Url類來方便的生成實(shí)際的URL地址(路由地址),針對(duì)上面的路由規(guī)則,我們可以用下面的方式生成URL地址。

// 輸出 blog/thinkphp
Url::build('blog/read', 'name=thinkphp');
Url::build('blog/read', ['name' => 'thinkphp']);
// 輸出 blog/5
Url::build('blog/get', 'id=5');
Url::build('blog/get', ['id' => 5]);
// 輸出 blog/2015/05
Url::build('blog/archive', 'year=2015&month=05');
Url::build('blog/archive', ['year' => '2015', 'month' => '05']);

[注意]build方法的第一個(gè)參數(shù)使用路由定義中的完整路由地址

還可以使用系統(tǒng)提供的助手函數(shù)url來簡(jiǎn)化

url('blog/read', 'name=thinkphp');
// 等效于
Url::build('blog/read', 'name=thinkphp');

通常在模板文件中輸出的話,可以使用助手函數(shù),例如:

{:url('blog/read', 'name=thinkphp')}

如果我們的路由規(guī)則發(fā)生調(diào)整,生成的URL地址會(huì)自動(dòng)變化

如果你配置了url_html_suffix參數(shù)的話,生成的URL地址會(huì)帶上后綴,例如:

'url_html_suffix'  => 'html',

那么生成的URL地址 類似

blog/thinkphp.html 
blog/2015/05.html

如果你的URL地址全部采用路由方式定義,也可以直接使用路由規(guī)則來定義URL生成,例如:

url('/blog/thinkphp');
Url::build('/blog/8');
Url::build('/blog/archive/2015/05');

生成方法的第一個(gè)參數(shù)一定要和路由定義的路由地址保持一致,如果你的路由地址比較特殊,例如使用閉包定義的話,則需要手動(dòng)給路由指定標(biāo)識(shí),例如: 

// 添加hello路由標(biāo)識(shí)
Route::rule(['hello','hello/:name'], function($name){
  return 'Hello,'.$name;
});
// 根據(jù)路由標(biāo)識(shí)快速生成URL
Url::build('hello', 'name=thinkphp');
// 或者使用
Url::build('hello', ['name' => 'thinkphp']);

關(guān)于怎么在thinkphp5中實(shí)現(xiàn)一個(gè)路由功能就分享到這里了,希望以上內(nèi)容可以對(duì)大家有一定的幫助,可以學(xué)到更多知識(shí)。如果覺得文章不錯(cuò),可以把它分享出去讓更多的人看到。

向AI問一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如果涉及侵權(quán)請(qǐng)聯(lián)系站長(zhǎng)郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI