13. 理解中间件Middleware - 从零开始学Laravel

   2016-12-01 0
核心提示:什么是中间件Middleware?如果是第一次接触,可能一时间会不太明白. 我们先想一下一个请求的生命周期,通常是发起了一个Http请求,服务器处理这个请求,然后对请求进行返回(Response).在服务器接收到请求,然后去执行业务逻辑之前,通常会需要判断这个请求是

什么是中间件Middleware?如果是第一次接触,可能一时间会不太明白. 我们先想一下一个请求的生命周期,通常是发起了一个Http请求,服务器处理这个请求,然后对请求进行返回(Response).

在服务器接收到请求,然后去执行业务逻辑之前,通常会需要判断这个请求是否是符合业务逻辑要求的,比如说,你的网站正在处于维护中,那么任何请求过来都不会让其进入到真正的业务逻辑,都是会直接返回503给用户,比如说我们之前说的用户发表评论的请求,当用户要发表一条评论,用户必须是处于登录过的状态的,如果发表评论的请求是未登录用户发出的,那么在请求进入业务逻辑前,就会返回未登录(403)。

像这样的情况会有很多,中间件做的就是这些事情,那么中间件就是处于路由和控制器之间的东西,当一条路由进来,先通过中间件进行判断,如果OK的,那么再进入控制器执行业务逻辑,如果不OK,那么就直接返回像503,403,404这些东东了。不知道这么说,大家能否理解。能感觉到意思就行,之后我们再通过代码来演示一下中间件。

再打个比方,中间件就像一层层的洋葱皮,请求需要通过这些洋葱皮,才能达到洋葱的中心(控制器中的具体业务逻辑),一层洋葱皮就是一个中间件,所以对于一个请求,你可以使用很多个中间件。

因为有了中间件,你的控制器就能专注于业务逻辑,不用去考虑其他了,这样一来,控制器会干净很多。

那我们的中间件都放在哪里呢? 在Laravel中给我们使用的有两个 Kernel.php 文件,一个位于 app/Http/Kernel.php , 另一个位于 app/Console/Kernel.php , 前面这个是用来处于web端Http请求的,后一个是用于命令行请求的,如我们经常使用的 php artisan ... 这些就是命令行的请求. 我们现在就看看关于web端Http请求的 Kernel.php 文件.

Kernel 类继承自 Illuminate\Foundation\Http\Kernel , 我们暂时不用管这个基类的别的功能,我们查看这个类,发现里面有这样的三个属性:

protected $middleware = [];

    protected $middlewareGroups = [];

    protected $routeMiddleware = [];

我们要做的事情就是覆写这三个属性,我们回头来看 app/Http/Kernel.php

<?php

namespace App\Http;

use Illuminate\Foundation\Http\Kernel as HttpKernel;

class Kernel extends HttpKernel
{
    // 这是一个全局的中间件,所有的路由都要经过这个中间件,无论你的路由来自前段的,还是来自api的。
    protected $middleware = [
        \Illuminate\Foundation\Http\Middleware\CheckForMaintenanceMode::class,
    ];

    // 中间件群组,网站上的请求都要经过web中间件群组,如果写给api的路由,则经过api群组
    protected $middlewareGroups = [
        'web' => [
            \App\Http\Middleware\EncryptCookies::class,
            \Illuminate\Cookie\Middleware\AddQueuedCookiesToResponse::class,
            \Illuminate\Session\Middleware\StartSession::class,
            \Illuminate\View\Middleware\ShareErrorsFromSession::class,
            \App\Http\Middleware\VerifyCsrfToken::class,
        ],

        'api' => [
            'throttle:60,1',
        ],
    ];
    
    // 非全局的中间件,单独给某些路由指定下面的中间件。
    protected $routeMiddleware = [
        'auth' => \App\Http\Middleware\Authenticate::class,
        'auth.basic' => \Illuminate\Auth\Middleware\AuthenticateWithBasicAuth::class,
        'can' => \Illuminate\Foundation\Http\Middleware\Authorize::class,
        'guest' => \App\Http\Middleware\RedirectIfAuthenticated::class,
        'throttle' => \Illuminate\Routing\Middleware\ThrottleRequests::class,
    ];
}

上面三个属性各代表了什么意思呢?我们先简单的了解下,先请看上面代码中的注解。然后我们再细致的来谈谈,我们先看中间件群组:

protected $middlewareGroups = [
        'web' => [
            \App\Http\Middleware\EncryptCookies::class,
            \Illuminate\Cookie\Middleware\AddQueuedCookiesToResponse::class,
            \Illuminate\Session\Middleware\StartSession::class,
            \Illuminate\View\Middleware\ShareErrorsFromSession::class,
            \App\Http\Middleware\VerifyCsrfToken::class,
        ],

        'api' => [
            'throttle:60,1',
        ],
    ];

我们通过代码来解释,中间件是位于请求到达控制器具体业务层之间的东西,那么我们就先打开路由文件routes.php:

Route::get('/', function () {
    return view('welcome');
});

Route::auth();

Route::get('/home', 'HomeController@index');

现在我们的路由文件的内容如上,因为我们现在是5.2.45版本(使用Laravel安装工具安装,会安装最新版的),在这之前我们这些写可不行,比如说在5.2.22版本中,我们需要这么写,当然了现在的版本按下面这样写那肯定也是对的,只不过现在的版本你不写web中间健群组,默认就当你已经使用的是web中间健群组。

Route::group(['middleware' => ['web']], function () {
    Route::get('/', function () {
        return view('welcome');
    });

    Route::auth();

    Route::get('/home', 'HomeController@index');
});

上面的意思就是, [web] 里面的路由一定要经过 web 中所有的中间件:

'web' => [
    \App\Http\Middleware\EncryptCookies::class,
    \Illuminate\Cookie\Middleware\AddQueuedCookiesToResponse::class,
    \Illuminate\Session\Middleware\StartSession::class,
    \Illuminate\View\Middleware\ShareErrorsFromSession::class,
    \App\Http\Middleware\VerifyCsrfToken::class,
],

我们看上面的中间件,我们之前就用到过两个,我们看 \Illuminate\View\Middleware\ShareErrorsFromSession::class, 这个就是我们验证表单错误时候将错误信息放入 $errors 中的中间件,还记得 $errors 吧,在来看 \App\Http\Middleware\VerifyCsrfToken::class, 这个就是 csrf 攻击的中间件,我们之前使用表单发生post的时候,如果没有添加上 {{csrf_field()}} , 那就通不过这个中间件了。大家应该也还记得。

我们来看下这个 \App\Http\Middleware\VerifyCsrfToken.php 文件的内容:

namespace App\Http\Middleware;

use Illuminate\Foundation\Http\Middleware\VerifyCsrfToken as BaseVerifier;

class VerifyCsrfToken extends BaseVerifier
{
    /**
     * The URIs that should be excluded from CSRF verification.
     *
     * @var array
     */
    protected $except = [
        //
    ];
}

这里面只有一个属性,当你希望有写路由不用通过这个中间件,你可以写在这里,那么这些路由就不会经过VerifyCsrfToken的审核了。

我们要看下该类的父类: Illuminate\Foundation\Http\Middleware\VerifyCsrfToken , 我们看handle()方法, 每一个中间件都会这个 handle() 方法,

public function handle($request, Closure $next)
    {
        // 如果是下面这些条件,那么将返回加入到cookie,然后让请求继续执行下去:$next($request)
    
        if (
            $this->isReading($request) ||
            $this->runningUnitTests() ||
            $this->shouldPassThrough($request) ||
            $this->tokensMatch($request)
        ) {
            return $this->addCookieToResponse($request, $next($request));
        }
        
        // 不符合上面的条件,那请求终止执行,抛出Token不匹配异常

        throw new TokenMismatchException;
    }

我们再来看下 \App\Http\Middleware\Authenticate.php

 
标签: 中间件 Laravel
反对 0举报 0 评论 0
 

免责声明:本文仅代表作者个人观点,与乐学笔记(本网)无关。其原创性以及文中陈述文字和内容未经本站证实,对本文以及其中全部或者部分内容、文字的真实性、完整性、及时性本站不作任何保证或承诺,请读者仅作参考,并请自行核实相关内容。
    本网站有部分内容均转载自其它媒体,转载目的在于传递更多信息,并不代表本网赞同其观点和对其真实性负责,若因作品内容、知识产权、版权和其他问题,请及时提供相关证明等材料并与我们留言联系,本网站将在规定时间内给予删除等相关处理.

  • bloom-server 基于 rust 编写的 rest api cache 中间件
    bloom-server 基于 rust 编写的 rest api cache
    bloom-server 基于 rust 编写的 rest api cache 中间件,他位于lb 与api worker 之间,使用redis 作为缓存内容存储, 我们需要做的就是配置proxy,同时他使用基于share 的概念,进行cache 的分布存储,包含了请求端口(proxy,访问数据) 以及cache 控制端口(
    03-08
  • Delphi Sysem.JSON 链式写法(转全能中间件)
    Delphi Sysem.JSON 链式写法(转全能中间件)
     链式写法有很多优点:连贯、语意集中、简洁、一气呵成、可读性强。比如要把 3.1415926 中的 59 提取为一个整数:Pi.ToString().Substring(5,2).ToInteger() 轻车熟路的感觉,让人觉得,哇!好帅的写法,牛x。      以前以为只有 TJSONBuilder 才支持链式
    02-09
  • nodeJS编写自己的中间件及示例返回字段由下划线改为驼峰的中间件
    nodeJS编写自己的中间件及示例返回字段由下划线
      Koa 是一个由 Express 原班人马打造的新的 web 框架,Koa 本身并没有捆绑任何中间件,只提供了应用(Application)、上下文(Context)、请求(Request)、响应(Response)四个模块。原本 Express 中的路由(Router)模块已经被移除,改为通过中间件的方
    02-09
  • 深入了解Go语言中web框架的中间件运行机制
    深入了解Go语言中web框架的中间件运行机制
    目录一、中间件的基本使用1.1 iris框架中间件的使用1.2 gin框架中使用中间件1.3 echo框架中使用中间件示例二、中间件的实现2.1 iris中间件实现2.2 gin中间件的实现2.3 echo框架中间件的实现三、中间件的运行机制3.1 iris中间件的运行机制3.2 gin中间件运行机
  • php中&&和||逻辑运算符怎么使用 php中间件
    本篇内容主要讲解“php中和||逻辑运算符怎么使用”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“php中和||逻辑运算符怎么使用”吧!一. 和||常规用法学过编程的朋友都知道条件判断吧,php中通常用if... elseif
    02-08 php
  • ASP.NET Core自定义中间件的方式
    ASP.NET Core自定义中间件的方式
    ASP.NET Core应用本质上,其实就是由若干个中间件构建成的请求处理管道。管道相当于一个故事的框架,而中间件就相当于故事中的某些情节。同一个故事框架采用不同的情节拼凑,最终会体现出不同风格的故事。而我们的ASP.NET Core应用也正是如此,同一管道采用不
    02-08
  • ASP.NET Core实现中间件的几种方式
    ASP.NET Core实现中间件的几种方式
    前言ASP.NET Core 中 HTTP 管道使用中间件组合处理的方式,换句人话来说,对于写代码的人而言,一切皆中间件.业务逻辑/数据访问/等等一切都需要以中间件的方式来呈现.那么我们必须学会如何实现自定义中间件 这里划重点,必考这里我们介绍下中间件的几种实现方式.
  • 前端需知nodejs express中间件使用及定义详解
    目录引言中间件的定义中间件的使用总结引言在上一节中我们简单了解了express的路由,本文主要分析下express的另一个核心功能中间件的使用。路由是express整个框架能有如今广泛使用和发展的基础,那中间件就是其发展的养分。正是express丰富的中间件才节省了我
  • nodejs关于中间件解析 nodejs 中间层
    nodejs关于中间件解析 nodejs 中间层
    目录nodejs 中间件中间件的编写中间件的使用nodejs制作自定义中间件总结nodejs 中间件今天我们来了解一下 nodejs 中间件,中间件可以说是 node 中最重要也是最核心的地方。它最重要的思想便是 洋葱模型 ,从外到内依次执行中间件。说了半天中间件到底有什么作
  • nodeJs编写错误处理中间件问题 nodejs 消息中间
    目录nodeJs编写错误处理中间件Connect中间件之错误处理中间件范例多个错误处理中间件组件nodeJs编写错误处理中间件app.use(async(ctx, next) = {try {await next();} catch (err) {console.log(err, "错误机制");ctx.status = err.status || err.statusCode;c
点击排行