HTTP 控制器


1、简介

将所有的请求处理逻辑都放在单个routes.php中肯定是不合理的,你也许还希望使用控制器类组织管理这些行为。控制器可以将相关的HTTP请求封装到一个类中进行处理。通常控制器存放在app/Http/Controllers目录中。

2、基本控制器

下面是一个基本控制器类的例子。所有的Laravel控制器应该继承自Laravel自带的控制器基类Controller

  1. <?php
  2. namespace App\Http\Controllers;
  3. use App\User;
  4. use App\Http\Controllers\Controller;
  5. class UserController extends Controller
  6. {
  7. /**
  8. * 为指定用户显示详情
  9. *
  10. * @param int $id
  11. * @return Response
  12. */
  13. public function showProfile($id)
  14. {
  15. return view('user.profile', ['user' => User::findOrFail($id)]);
  16. }
  17. }

我们可以像这样定义指向该控制器动作的路由:

  1. Route::get('user/{id}', 'UserController@showProfile');

现在,如果一个请求匹配上面的路由URI,UserControllershowProfile方法就会被执行。当然,路由参数也会被传递给这个方法。

2.1 控制器&命名空间

你应该注意到我们在定义控制器路由的时候没有指定完整的控制器命名空间,而只是定义了App\Http\Controllers之后的部分。默认情况下,RouteServiceProvider将会在一个路由分组中载入routes.php文件,并且该路由分组指定定了分组中路由控制器所在的命名空间。

如果你在App\Http\Controllers目录下选择使用PHP命名空间嵌套或组织控制器,只需要使用相对于App\Http\Controllers命名空间的指定类名即可。因此,如果你的完整控制器类是App\Http\Controllers\Photos\AdminController,你可以像这样注册路由:

  1. Route::get('foo', 'Photos\AdminController@method');

2.2 命名控制器路由

和闭包路由一样,可以指定控制器路由的名字:

  1. Route::get('foo', ['uses' => 'FooController@method', 'as' => 'name']);

设置好控制器路由后,就可以使用帮助函数action很方便地为控制器动作生成对应的URL:

  1. $url = action('FooController@method');

你还可以使用帮助函数route来为已命名的控制器路由生成对应的URL:

  1. $url = route('name');

3、控制器中间件

中间件可以像这样分配给控制器路由:

  1. Route::get('profile', [
  2. 'middleware' => 'auth',
  3. 'uses' => 'UserController@showProfile'
  4. ]);

但是,将中间件放在控制器构造函数中更方便,在控制器的构造函数中使用middleware方法你可以很轻松的分配中间件给该控制器。你甚至可以限定该中间件应用到该控制器类的指定方法:

  1. class UserController extends Controller
  2. {
  3. /**
  4. * 实例化一个新的 UserController 实例
  5. *
  6. * @return void
  7. */
  8. public function __construct()
  9. {
  10. $this->middleware('auth');
  11. $this->middleware('log', ['only' => ['fooAction', 'barAction']]);
  12. $this->middleware('subscribed', ['except' => ['fooAction', 'barAction']]);
  13. }
  14. }

4、RESTful资源控制器

Laravel的资源控制器使得构建围绕资源的RESTful控制器变得毫无痛苦,例如,你可能想要在应用中创建一个控制器,用于处理关于图片存储的HTTP请求,使用Artisan命令make:controller,我们可以快速创建这样的控制器:

  1. php artisan make:controller PhotoController

该Artisan命令将会生成一个控制器文件app/Http/Controllers/PhotoController.php,这个控制器包含了每一个资源操作对应的方法。

接下来,可以为该控制器注册一个资源路由:

  1. Route::resource('photo', 'PhotoController');

这个路由声明包含了处理图片资源RESTful动作的多个路由,相应地,Artisan生成的控制器也已经为这些动作设置了对应的处理方法。

4.1 资源控制器处理的动作

方法 路径 动作 路由名称
GET /photo index photo.index
GET /photo/create create photo.create
POST /photo store photo.store
GET /photo/{photo} show photo.show
GET /photo/{photo}/edit edit photo.edit
PUT/PATCH /photo/{photo} update photo.update
DELETE /photo/{photo} destroy photo.destroy

4.2 只定义部分资源路由

声明资源路由时可以指定该路由处理的动作子集:

  1. Route::resource('photo', 'PhotoController',
  2. ['only' => ['index', 'show']]);
  3. Route::resource('photo', 'PhotoController',
  4. ['except' => ['create', 'store', 'update', 'destroy']]);

4.3 命名资源路由

默认情况下,所有资源控制器动作都有一个路由名称,然而,我们可以通过传入names数组来覆盖这些默认的名字:

  1. Route::resource('photo', 'PhotoController',
  2. ['names' => ['create' => 'photo.build']]);

4.4 嵌套资源

有时候我们需要定义路由到“嵌套”资源。例如,一个图片资源可能拥有多条“评论”,要“嵌套”资源控制器,在路由声明中使用“.”号即可:

  1. Route::resource('photos.comments', 'PhotoCommentController');

该路由将注册一个嵌套的资源,使用URL访问方式如下:
photos/{photos}/comments/{comments}.

  1. <?php
  2. namespace App\Http\Controllers;
  3. use App\Http\Controllers\Controller;
  4. class PhotoCommentController extends Controller
  5. {
  6. /**
  7. * 显示指定照片评论
  8. *
  9. * @param int $photoId
  10. * @param int $commentId
  11. * @return Response
  12. * @translator http://laravelacademy.org
  13. */
  14. public function show($photoId, $commentId)
  15. {
  16. //
  17. }
  18. }

4.5 补充资源控制器

如果有必要在默认资源路由之外添加额外的路由到资源控制器,应该在调用Route::resource之前定义这些路由;否则,通过resource方法定义的路由可能无意中优先于补充的额外路由:

  1. Route::get('photos/popular', 'PhotoController@method');
  2. Route::resource('photos', 'PhotoController');

扩展阅读:实例教程——创建RESTFul风格控制器实现文章增删改查

5、隐式控制器

Laravel允许你只定义一个路由即可访问控制器类中的所有动作,首先,使用Route::controller方法定义一个路由,该controller方法接收两个参数,第一个参数是控制器处理的baseURI,第二个参数是控制器的类名:

  1. Route::controller('users', 'UserController');

接下来,添加方法到控制器,方法名应该以HTTP请求方法开头:

  1. <?php
  2. namespace App\Http\Controllers;
  3. class UserController extends Controller
  4. {
  5. /**
  6. * 响应 GET /users 请求
  7. */
  8. public function getIndex()
  9. {
  10. //
  11. }
  12. /**
  13. * 响应 GET /users/show/1 请求
  14. */
  15. public function getShow($id)
  16. {
  17. //
  18. }
  19. /**
  20. * 响应 GET /users/admin-profile 请求
  21. */
  22. public function getAdminProfile()
  23. {
  24. //
  25. }
  26. /**
  27. * 响应 POST /users/profile 请求
  28. */
  29. public function postProfile()
  30. {
  31. //
  32. }
  33. }

在上例中可以看到,getIndex方法将会在访问控制器处理的默认URI——users时被调用。

5.1 分配路由名称

如果你想要命名该控制器中的一些路由,可以将一个名称数组作为第三个参数传递到该controller方法:

  1. Route::controller('users', 'UserController', [
  2. 'getShow' => 'user.show',
  3. ]);

6、依赖注入 & 控制器

6.1 构造函数注入

Laravel使用服务容器解析所有的Laravel控制器,因此,可以在控制器的构造函数中类型声明任何依赖,这些依赖会被自动解析并注入到控制器实例中:

  1. <?php
  2. namespace App\Http\Controllers;
  3. use Illuminate\Routing\Controller;
  4. use App\Repositories\UserRepository;
  5. class UserController extends Controller
  6. {
  7. /**
  8. * The user repository instance.
  9. */
  10. protected $users;
  11. /**
  12. * 创建新的控制器实例
  13. *
  14. * @param UserRepository $users
  15. * @return void
  16. */
  17. public function __construct(UserRepository $users)
  18. {
  19. $this->users = $users;
  20. }
  21. }

当然,你还可以类型提示任何Laravel契约,如果容器可以解析,就可以进行类型提示。

6.2 方法注入

除了构造函数注入之外,还可以在控制器的动作方法中进行依赖的类型提示,例如,我们可以在某个方法中类型提示Illuminate\Http\Request实例:

  1. <?php
  2. namespace App\Http\Controllers;
  3. use Illuminate\Http\Request;
  4. use Illuminate\Routing\Controller;
  5. class UserController extends Controller
  6. {
  7. /**
  8. * 存储新用户
  9. *
  10. * @param Request $request
  11. * @return Response
  12. */
  13. public function store(Request $request)
  14. {
  15. $name = $request->input('name');
  16. //
  17. }
  18. }

如果控制器方法期望输入路由参数,只需要将路由参数放到其他依赖之后,例如,如果你的路由定义如下:

  1. Route::put('user/{id}', 'UserController@update');

你需要通过定义控制器方法如下所示来类型提示Illuminate\Http\Request并访问路由参数id

  1. <?php
  2. namespace App\Http\Controllers;
  3. use Illuminate\Http\Request;
  4. use Illuminate\Routing\Controller;
  5. class UserController extends Controller
  6. {
  7. /**
  8. * 更新指定用户
  9. *
  10. * @param Request $request
  11. * @param int $id
  12. * @return Response
  13. * @translator http://laravelacademy.org
  14. */
  15. public function update(Request $request, $id)
  16. {
  17. //
  18. }
  19. }

7、路由缓存

如果你的应用完全基于路由使用控制器,可以使用Laravel的路由缓存,使用路由缓存将会极大减少注册所有应用路由所花费的时间开销,在某些案例中,路由注册速度甚至能提高100倍!想要生成路由缓存,只需执行Artisan命令route:cache:

  1. php artisan route:cache

就这么简单!你的缓存路由文件现在取代app/Http/routes.php文件被使用,记住,如果你添加新的路由需要重新生成路由缓存。因此,只有在项目开发阶段你才会运行route:cache命令。

想要移除缓存路由文件,使用route:clear命令即可:

  1. php artisan route:clear