Laravel 文档阅读:路由

基础路由


在 Laravel 中要简单定义一个路由非常方便,传一个 URI 和 闭包即可。

Route::get('foo', function(){
    return "Hello World";
});

默认路由文件

Laravel 中所有的路由文件定义在 routes 目录下,这个目录下的内容会自动被框架加载。在 routes/web.php 文件中定义的是 Web 接口路由,这些路由默认使用了 web 中间件组过率,它提供了 Session 会话和 CSRF 保护功能。
routes/api.php 文件用来定义 API 接口路由,这些路由都是无状态的,默认使用了 api 中间件过滤。
许多项目,基本上都是以 routes/web.php 为起点进行开发的,它能实现快速开发一个项目的需要。在 routes/web.php 中定义的路由在浏览器中是可以直接访问的到的。例如,在浏览器输入 http://your-app.dev/user 地址来访问下面的路由:

Route::get('/user', 'UsersController@index);

定义在 routes/web.php 中的路由被嵌套在了一个路由组里,这是在 RouteServiceProvider 中设定的。这个组里的路由都使用了 /api 这个前缀,这样在你定义路由的时候,就不必再添加这个前缀了。如果你不想使用 /api 这个前缀的话,那么可以在 RouteServiceProvider 中修改。

/**
 * Define the "api" routes for the application.
 *
 * These routes are typically stateless.
 *
 * @return void
 */
protected function mapApiRoutes()
{
     Route::prefix('api')
            ->middleware('api')
            ->namespace($this->namespace)
            ->group(base_path('routes/api.php));
}

可用的路由方法

每一个 HTTP 请求类型都有对应的路由方法可供使用:

Route::get($uri, $callback);
Route::post($uri, $callback);
Route::put($uri, $callback);
Route::patch($uri, $callback);
Route::delete($uri, $callback);
Route::options($uri, $callback);

有时一个路由需要能够匹配多个请求类型,这时可以用 match方法:

Route::match(['get', 'post'], '/', function(){
     //
});

如果希望一个路由能够匹配所有请求类型的话,使用 any 方法:

Route::any('foo', function(){
     //
});

CSRF 保护

使用 Route::postRoute::putRoute::patchRoute::delete 定义的路由,要请求访问他们的 HTNL表单都要传递一个 CSRF token ,否则请求会被拒绝。你可以在 CSRF 文档里阅读更多这方面的内容。

<form method="post" action="profile">
     {{ csrf_filed() }}
</form>

重定向路由

如果你要将一个路由重定向另一个路由,可以使用 redirect 方法,这就节省一个定义一个完整路由或者控制器来操作简单的重定向带来的一些麻烦:

Route::redirect('here', 'there', 301)

视图路由

如果你只是要简单的返回一个视图,可以使用 Route::view 方法,类似 Route::redirect() 方法,也节省了一些麻烦。view 方法的第一个参数是 URI, 第二个参数是视图名。另外,也可以使用可选的第三个数组类型参数向视图传递数据:

Route::view('/welcome', 'welcome');
Route::view('/welcome', 'welcome', [ 'name' => 'Dave']);

路由参数


必须参数

有时需要参数捕获路由 URI 片段,得到我们要用的参数值。例如,我么们从一个路由的 URL 里捕获用户的 ID ,你可以这样定义一个参数:

Route::get('user/{id}', function($id){
      return 'User'.$id;
})

你可以按照需要在路由里定义多个路由参数。

Route::get('posts/{post}/comments/{comment}, function($postId, $commentId){
      //
});

可选参数

上面的路由参数,在 HTTP 请求中都是必须传递的,如果你需要路由参数是可选的路由参数,那么就需要在参数名后面放上一个 ? 标识,表示这是可选参数,不过这是要保证对应位置上的 “形参必须要有默认值”.

Route::get('user/{name?}', function($name = null){
     return $name;
});

Route::get('user/{name}', function($name = 'John'){
     return $name;
});

正则表达式约束

可以对路由参数做各种限制的,这时要用到路由实例的 where 方法。 where 方法的第一个参数是路由参数名,第二个参数就是一个正则表达式(规定参数格式):

Route::get('user/{name}', function($name){
     //
})->where('name', '[A-Za-z]+');

Route::get('user/{id}', function($id){
     //
})->where('id', '[0-9]+');

Route::get('user/{id}/{name}', function($id, $name){
     //
})->where(['id' => '[0-9]+', 'name' => '[A-Za-z]+' ])

全局约束

如果在项目里,一个路由参数名的规则都是一样的,这时就可以用 pattern 方法定义全局约束,具体是在 RouteServiceProviderboot 里使用 pattern 方法定义规则。

/**
 * Define your route model bindings, pattern filters, etc.
 *
 * @return void
 */
public function boot()
{
    Route::pattern('id','[0-9]+');
    parent::boot();
}

全局约束规则定义好之后,所有路由中匹配该参数名的路由参数值都受该规则限制。

Route::get('user/{id}', function($id){
   // Only executed if {id} is numeric.....
})

命名路由


命名路由就是给路由起一个名字,有了名字后,我们就可以用这个名字来生成访问这个路由的 URL 地址了。在定义路由的时候,链式调用 name 方法,就给路由起了一个名字:

Route::get('user/profile', function(){
   //
})->name('profile');

也可以为基于控制器的 action 的路由起名:

Route::get('user/profile','UsersController@profile')->name('profile');

生成命名路由的 URL

访问命名路由,使用全局 route 辅助函数,这个函数会生成访问这个路由的 URL 地址:

//Generating URLs
$url = route('profile');

// Generating redirects...
return redirect()->route('profile');

如果命名路由中包含参数,那么调用 route 函数的时候,传递第二个参数来指定路由参数值。

Route::get('user/{id}/profile',function($id){
  //
})->name('profile');

$url = route('profile', ['id' => 1]);

检查当前路由

如果你想知道,当前请求是否发送到一个特定的命名路由上的,用 Route 实例的 named 方法就可以做到。例如,我们在一个路由中间件里定义了这个逻辑:

/**
 * Handle an incoming request.
 *
 * @param  \Illuminate\Http\Request  $request
 * @param  \Closure  $next
 * @return mixed
 */
public function handle($request, Closure $next)
{
   //如果当前请求是发往名字是 `profile` 的路由的,就会执行 if 里面的逻辑
     if( $request->route()->named('profiled')){
         //
      }

return $next($request);
}

路由组


当我们定义的路由很多的时侯,就会发现有些路由是有一些共同特征的,比如用了同一个中间件、有相同的命名空间、有一样的 URL 前缀地址等等。这时为了方便,我们就不要给每个路由重复、单独地定义这些内容了,使用 Route::group 方法就可以轻松解决这个麻烦,Route::group 方法让具有共同特性的路由放到一个组里面,我们称这个组叫 「路由组」。

中间件

给一个路由组应用中间件,是在 group 方法之前调用 middleware 方法,中间件会按照中间件在数组里出现的顺序依次执行:

Route::middleware(['first', 'second'])->group(function(){
     Route::get('/', function(){
        // User first & second middleware;
     })

     Route::get('user/profile',function(){
        // User first & second middleware
     })
});

命名空间

另一个路由组的常用用处是使用 namespace 方法为添加统一的命名空间:

Route::namespace('Admin')->group(function(){
     //Controllers Within The "App\Http\Controllers\Admin" Namespace
})

需要注意的是:在 RouteServiceProvider 中已经为 routes/web.phproutes/api.php 文件中定义的路又添加了 App\Http\Controllers 的命名空间前缀。所以在定义路由时,你只需要指定 App\Http\Controllers 命名空间的后面部分就可以了。

子域路由

路由组可以用于处理子域地址,在子域上也可以使用参数,路由能捕获子域地址里的参数值。在 group 方法前调用 domain 方法就可以使用这个功能了:

Route::domain('{account}.myapp.com')->group(function(){
     Route::get('user/{id}', function($account, $id){
         //
     });
});

路由前缀

prefix 方法用来给路由组添加 URI 前缀。例如,我们给所有路由 URL 添加 admin 前缀:

Route::prefix('admin')->group(function(){
       Route::get('users', function(){
            //Matches The "/admin/users" URL
       });
});

路由模型绑定


我们通常在路由或者控制器 action 中获取一个模型 ID 值,然后通过这个 ID 找到对应的模型实例,Laravel 路由模型绑定让这个过程更加简单,它会自动在你的路由中注入模型实例。例如,传递一个用户 ID,得到对应的用户实例。

隐形绑定

Laravel 会自动解析路由或控制 action 中的形参变量,但前提是该变量名与路由参数名一样、且有类型(Eloquent Model )提示的

Route::get('api/users/{user}', function(App\User $user){
       return $user->email;
});

在这里,$user 变量的类型是 App\User 这个 Eloquent Model,并且这个变量名是和 URL 参数一样的,所以 Laravel 会自动将拥有该 ID 的 App\User 实例对象赋值给 $user 变量。如果实例对象在数据库中未找到,就会抛出一个 404 响应。

自定义模型在路由中主键名

如果要让路由绑定从数据库中获得模型实例使用的默认字段不是 id, 而是其他字段,这时候就需要在 Eloquent Model 中重写 getRouteKeyName 方法:

/**
 * Get the route key for the model.
 *
 * @return string
 */
public function getRouteKeyName()
{
        return 'slug';
}

精确绑定

RouteServoceProvider 类的 boot 方法中,使用 Route::model 方法为特定的路由参数名绑定 Eloquent Model 类:

public function boot()
{
     parent::boot();
     Route::model('user', App\User::class);
}

接下来在路由中使用参数 {user}

Route::get('profile/{user}', function(){
       //
})

我们把所有路由参数是 {user} 的都绑定给了 App\User 模型,最后对应的 User 模型实例就会注入到路由中。例如, profile/1 的请求地址最终会从数据库中获得 ID 为 1 的那个用户数据。如果在数据库中没有找到匹配的用户,就会自动返回 404 响应。

自定义解析逻辑

如果希望解析路由参数时,使用自己的一套更加个性化的逻辑, 使用 Route::bind 方法就可以了:

public function boot()
{
    parent::boot();
    Route::bind('user', function($value){
        return App\User::where('name',$value)->first();
    });
}

表单方法伪造


HTML 不支持 PUTPATCHDELETE 方式的 HTTP请求。 为了能与后台的 PUTPATCHDELETE 类型的路由匹配,就需要添加一个名为 _method 的隐藏表单域。_method 这个隐藏表单域的值会被 Laravel 看做 HTTP 请求的标识:

<form action="/foo/bar" method="POST">
      <input type="hidden" name="method" value="PUT">
      <input type="hidden" name="_token" value="{{ csrf_token }}">

</form>

你也可以使用 method_field 辅助函数产生这个 method 隐藏表单域:

{{ method_field('PUT') }}

访问当前路由


使用 Route 门面的 currentcurrentRouteNamecurrentRouteAction 方法获得处理当前请求的路由信息:

$route = Route::current();
$name = Route::currentRouteName();
$action = Route::currentRouteAction();

更多关于 Route 门面的底层类Route 实例
的所有可用方法请参考 API 文档。

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,922评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,591评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,546评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,467评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,553评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,580评论 1 293
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,588评论 3 414
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,334评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,780评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,092评论 2 330
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,270评论 1 344
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,925评论 5 338
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,573评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,194评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,437评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,154评论 2 366
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,127评论 2 352

推荐阅读更多精彩内容

  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,647评论 18 139
  • 路由 路由(routing)就是通过互联的网络把信息从源地址传输到目的地址的活动。路由发生在OSI网络参考模型中的...
    Dearmadman阅读 2,856评论 2 9
  • Laravel 学习交流 QQ 群:375462817 本文档前言Laravel 文档写的很好,只是新手看起来会有...
    Leonzai阅读 7,868评论 2 12
  • 22年12月更新:个人网站关停,如果仍旧对旧教程有兴趣参考 Github 的markdown内容[https://...
    tangyefei阅读 35,176评论 22 257
  • 0.1配置1.模板继承2.控制器3.git4.支付宝支付的流程5.路由6.中间件7.请求8.laravel 学习笔...
    云龙789阅读 806评论 0 5