Laravel 路由学习

Larvel路由实现原理和咱们平时说的路由器原理表面上差不多,主要是实现请求分派,用户通过各种各样的Url结合http请求,形成一个Request 然后这个Request被移转给路由器进行分派。路由器会将请求分派给路由本身或控制器,并执行任何特定路由的中间件。简单的说路由是Url和程序模块之间的一个映射。 下面我们从下面几个方面进行讲解

  • 基本路由
  • 多请求路由
  • 路由参数
  • 路由别名
  • 路由群组
  • 路由中输出视图

前言:

如果大家了解Laravel5.3版本以前的,路由文件默认是放在 根目录/app/Http/routes.php,如果业务复杂的时候那,需要自己进行路由分割策略。而5.3以后那Laravel 路由都在 根目录/routes 中的路由文件中定义,这些文件都由框架自动加载。在 routes/web.php 文件中定义你的 web 页面路由。这些路由都会应用 web 中间件组,其提供了诸如 Session 和 CSRF 保护等特性。定义在 routes/api.php 中的路由都是无状态的,并且会应用 api 中间件组。大多数的应用构建,都是以在 routes/web.php 文件定义路由开始的。

下面路由演示均是在www.xiaot123.com域名下访问,咱们简称xiaoT

1、基础路由

1.1基本路由实现

咱们先来个最简单最基础的路由,它是由一个Url中的字符 +闭包函数构成

Route::get('x', function () {
    return 'Hello XiaoT';
});

大家按照这个路由访问 xiaoT/x 页面上会出现Hello XiaoT字样即路由中return的字符串直接输出到了页面上。大家这里可以试试输出个数组试试^_^

1.2 常用的http请求方法所对应的路由 了解更多Http请求方法

Http请求方法 对应的路由写法
GET Route::get($uri, $callback);
POST Route::post($uri, $callback);
PUT Route::put($uri, $callback);
DELETE Route::delete($uri, $callback);
PATCH Route::patch($uri, $callback);
OPTIONS Route::options($uri, $callback);

当然上面的$callback闭包都可以换成控制器访问模式(控制器@方法名),如Route::get('/xiaoT', 'xiaoTController@info'); 标示访问xiaoTController控制器中的info方法,请求方式是GET

1.3 可以接受任何Http方法请求的路由

Route::any('foo', function () {
    retrun 'I can get all  Http method requests'
});

用途距离 :在开发微信相关如微信授权登陆回调时 和 微信支付回调时建议要用any方式的路由

1.3 资源路由

RESTful是一种设计思想、一种普遍接受的规范。我们的资源路由,和RESTful有着莫大的联系,要理解资源路由(和资源控制器),必须先了解RESTful。

REST这个词,是Roy Thomas Fielding在他2000年的博士论文中提出的。

Fielding是一个非常重要的人,他是HTTP协议(1.0版和1.1版)的主要设计者、Apache服务器软件的作者之一、Apache基金会的第一任主席。所以,他的这篇论文一经发表,就引起了关注,并且立即对互联网开发产生了深远的影响。

Fielding将他对互联网软件的架构原则,定名为REST,即Representational State Transfer的缩写。我对这个词组的翻译是"表现层状态转化"。

如果一个架构符合REST原则,就称它为RESTful架构。

要理解RESTful架构,最好的方法就是去理解Representational State Transfer这个词组到底是什么意思,它的每一个词代表了什么涵义。如果你把这个名称搞懂了,也就不难体会REST是一种什么样的设计。

大家一定要阅读该文章,理解RESTful架构,文章十分清晰的讲述了RESTful,本文就不再重复熬述。

我们如果要在路由里定义一个资源控制器只需要一条:

Route::resource('xiaot', 'xiaotController');

那么它对应的路由访问规则如下:

请求方法 请求URI 对应的控制器方法 代表的意义
GET /xiaot index 索引单条数据/获取列表数据列表
GET /xiaot/create create 创建(显示表单)
POST /xiaot store 保存你创建的数据
GET /xiaot/{id} show 显示对应id的内容
GET /xiaot/{id}/edit edit 编辑(显示表单)
PUT/PATCH /xiaot/{id} save 保存你编辑的数据
GET /xiaot/{id} destroy 删除该条记录

说明: 当我GET方法访问地址xiaoT/xiaot,相当于访问控制器xiaotController的index方法。 当我POST方法访问地址xiaoT/xiaot,就会访问到xiaotController的store方法。 有了资源路由我们很容易实现一个符合RESTful架构的接口,这种很适合作为APP后端开发时使用。这种规范下,不但访问策略清晰易理解,更容易维护。也使你的架构更为合理和现代化。

2、多请求路由

有的时候你可能需要注册一个可响应多个 HTTP 方法的路由,这时你可以使用 match 方法,也可以使用 any 方法注册一个实现响应所有 HTTP 的请求的路由:关闭any前面讲了不在赘述

Route::match(['get', 'post'], '/', function () {
    //可以接受GET和POST请求方法
});

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

3、路由参数

3.1 必选路由参数

当然,有时我们需要在路由中捕获一些 URL 片段。例如,我们需要从 URL 中捕获用户的 ID ,我们可以这样定义路由参数:

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

也可以根据需要在路由中定义多个参数:

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

路由的参数通常都会被放在 {}内,并且参数名只能为字母,当运行路由时,参数会通过路由闭包来传递。

注意: 路由参数不能包含 - 字符。请用下划线 (_) 替换

3.2 可选路由参数

声明路由参数时,如需指定该参数为可选,可以在参数后面加上 ? 来实现,但是相应的变量必须有默认值:

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

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

3.3 正则表达式约束

你可以使用 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-z]+']);

3.4全局约束

如果你希望路由参数在全局范围内都遵循一个确定的正则表达式约束,则可以使用pattern 方法。你应该在 RouteServiceProviderboot 方法里定义这些模式:

/**
 * 定义你的路由模型绑定, pattern 过滤器等。
 *
 * @return void
 */
public function boot()
{
    Route::pattern('id', '[0-9]+');

    parent::boot();
}
Pattern 一旦被定义,便会自动应用到所有使用该参数名称的路由上:

Route::get('user/{id}', function ($id) {
    // 仅在 {id} 为数字时执行...
});

4、路由别名

为路由指定了名称后,我们可以使用全局辅助函数 route 来生成 URL 或者重定向到该条路由:

// 生成 URL...
$url = route('profile');

// 生成重定向...
return redirect()->route('profile');

如果是有定义参数的命名路由,可以把参数作为 route 函数的第二个参数传入,指定的参数将会自动插入到 URL 中对应的位置:

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

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

5、路由群组

路由组允许共享路由属性,例如中间件和命名空间等,我们没有必要为每个路由单独设置共有属性,共有属性会以数组的形式放到 Route::group 方法的第一个参数中。

5.1 中间件

要给路由组中定义的所有路由分配中间件,可以在路由组中使用 middleware 键,中间件将会依照列表内指定的顺序运行:

Route::group(['middleware' => 'auth'], function () {
    Route::get('/', function ()    {
        // 使用 `Auth` 中间件
    });

    Route::get('user/profile', function () {
        // 使用 `Auth` 中间件
    });
});

5.2 命名空间

另一个常见的例子是,为控制器组指定公共的 PHP 命名空间。这时使用 namespace 参数来指定组内所有控制器的公共命名空间:

Route::group(['namespace' => 'Admin'], function () {
    // 在 "App\Http\Controllers\Admin" 命名空间下的控制器
});
``
请记住,默认 `RouteServiceProvider` 会在命名空间组中引入你的路由文件,让你不用指定完整的 `App\Http\Controllers` 命名空间前缀就能注册控制器路由,因此,我们在定义的时候只需要指定命名空间 `App\Http\Controllers` 以后的部分。

#### 5.3  子域名路由
路由组也可以用作子域名的通配符,子域名可以像 `URI` 一样当作路由组的参数,因此允许把捕获的子域名一部分用于我们的路由或控制器。可以使用路由组属性的 `domain` 键声明子域名。
```PHP
Route::group(['domain' => '{account}.myapp.com'], function () {
    Route::get('user/{id}', function ($account, $id) {
        //
    });
});

5.4 路由前缀

通过路由组数组属性中的prefix 键可以给每个路由组中的路由加上指定的 URI 前缀,例如,我们可以给路由组中所有的URI 加上路由前缀 admin :

Route::group(['prefix' => 'admin'], function () {
    Route::get('users', function ()    {
        // 匹配包含 "/admin/users" 的 URL
    });
});

6、路由中输出视图

没什么好说的直接上代码

Route::get(‘view‘,function(){
    return view(‘viewname‘);
  });

如果有问题,可以留言给我哦