中间件,第一次听到这个名字感觉好陌生,这是个啥呀,第三方插件?好像不是。之前也没有遇到过这个玩意啊。
之前使用到的thinkphp5.0以及Yii2.0框架都是没有中间件这一说的。
去thinkphp官网查了一下,要到thinkphp5.1.6才开始支持中间件。实现的方式基本上就是仿照laravel。
一:那么什么时中间件呢:
HTTP 中间件提供了为过滤进入应用的 HTTP 请求提供了一套便利的机制。
例如,Laravel 内置了一个中间件来验证用户是否经过授权,如果用户没有经过授权,中间件会将用户重定向到登录页面,否则如果用户经过授权,中间件就会允许请求继续往前进入下一步操作。
当然,除了认证之外,中间件还可以被用来处理更多其它任务。比如:CORS 中间件可以用于为离开站点的响应添加合适的头(跨域);日志中间件可以记录所有进入站点的请求。
Laravel框架自带了一些中间件,包括维护模式、认证、CSRF 保护中间件等等。所有的中间件都位于 app/Http/Middleware 目录。
二:创建中间件
1:命令行创建
比如我要创建一个test中间件
php artisan make:middleware Test
2:手动创建
进入app/Http/Middleware目录,找一个你觉得看着舒服的中间件文件,复制,改名,进入文件,改类名。完成操作。
三:中间件的一些说明
我创建的中间件如下所示:
<?php
namespace App\Http\Middleware;
// 使用闭包类
use Closure;
class opLog
{
/**
* Handle an incoming request.
*
* @param \Illuminate\Http\Request $request
* @param \Closure $next
* @return mixed
*/
public function handle($request, Closure $next)
{
if($request->input('token') != 'guanchao.site')
{
return redirect()->to('https://guanchao.site');
}
return $next($request);
}
}
这里需要注意几个问题:
1:中间件的入口执行方法必须是handle方法。
2:第一个参数是Request对象,第二个参数是一个闭包。
3:中间件handle方法的返回值必须是一个Response对象。
4:中间件可以有第三个参数:
<?php
namespace App\Http\Middleware;
use Closure;
class CheckRole
{
/**
* 处理输入请求
* @param \Illuminate\Http\Request $request
* @param \Closure $next
* @param string $role
* @return mixed
* translator https://xueyuanjun.com
*/
public function handle($request, Closure $next, $role)
{
if (! $request->user()->hasRole($role)) {
// Redirect...
}
return $next($request);
}
}
在这个中间件中我们判断当前请求的token参数不等于guanchao.site的时候进行重定向处理。否则,请求将进一步传递到应用中。要让请求继续传递到应用程序中,只需使用 next 。
四:前/后置中间件
一个中间件是请求前还是请求后执行取决于中间件本身。比如,以下中间件会在请求处理前执行一些任务:
<?php
namespace App\Http\Middleware;
use Closure;
class BeforeMiddleware
{
public function handle($request, Closure $next)
{
// 执行动作
return $next($request);
}
}
而下面这个中间件则会在请求处理后执行其任务:
<?php
namespace App\Http\Middleware;
use Closure;
class AfterMiddleware
{
public function handle($request, Closure $next)
{
$response = $next($request);
// 执行动作
return $response;
}
}
五:注册中间件
中间件大概有三中吧注册方式middlewareGroup $routeMiddleware
试用之前提前说一下:调用自定义中间件的时候,需要在命名空间哪里use一下。
use App\Http\Middleware\opLog;
1:$middleware 全局生效
/**
* 全局中间件
*/
protected $middleware = [
opLog::class
];
2:$middlewareGroup 中间件组(针对不同的路由文件配置)
/**
* 中间件组
*/
protected $middlewareGroups = [
'web' => [
\App\Http\Middleware\VerifyCsrfToken::class,
\Illuminate\Routing\Middleware\SubstituteBindings::class,
],
'api' => [
'throttle:60,1',
\Illuminate\Routing\Middleware\SubstituteBindings::class,
],
];
3:$routeMiddleware 路由中间件
/**
* 路由中间件
*/
protected $routeMiddleware = [
‘opLog’=> opLog::class,
];
将中间件配置到这里,就可以在路由中调用这个中间件了,给指定路由调用。例:
Route::group(['middleware' => 'web'],function($route){
$route->get('hello/world',function(){});
$route->get('hello/php',function(){});
// 这样在访问这个这些路由的时候,就会执行中间件组 web 所对应的中间件!
});
Route::get('/xxx', xxxx\xxxxx@xxx)->middleware(opLog);
六:中间件排序
在某些特殊场景下,你可能需要中间件按照特定顺序执行,但是一旦中间件已经分配到指定路由就没法控制它们的执行顺序了。在这种情况下,你可以在 app/Http/Kernel.php 文件中通过 $middlewarePriority 属性来指定中间件的优先级:
/**
* The priority-sorted list of middleware.
*
* This forces non-global middleware to always be in the given order.
*
* @var array
*/
protected $middlewarePriority = [
\Illuminate\Session\Middleware\StartSession::class,
\Illuminate\View\Middleware\ShareErrorsFromSession::class,
\App\Http\Middleware\Authenticate::class,
\Illuminate\Session\Middleware\AuthenticateSession::class,
\Illuminate\Routing\Middleware\SubstituteBindings::class,
\Illuminate\Auth\Middleware\Authorize::class,
];
有好的建议,请在下方输入你的评论。
原文链接:https://guanchao.site/index/article/articledetail.html?artid=mo7k1WXD0