搜索
首页php框架Laravel详解如何使用ETag和条件标头进行缓存

Laravel API 性能优化:使用 ETag 和条件标头进行缓存

当写一个前后端分离的应用时,你必须得开始考虑前端客户端会给API提交什么样的请求,从后端再次获取数据,即使你只想要验证前端缓存是否能对添加的数据实时更新。根据以上的需求,你可以使用 ETag 头 和 conditional requests。

在这篇博文中,我会简单的概括一下 ETagIf-None-Match 和 If-Match 头是做什么的,然后再看看我是如何将这些应用到我们的package中的,这个软件包可以快速的将它实施到您的应用程序中。

是什么

让我们从这一切的核心内容开始,那就是 ETag 头。该头文件是表示其所在的确切状态下的响应主体的值。在很多情况下 ETag 的值将是内容的 hash 值,因为这是最容易生成和保证响应数据唯一性标识符的方法。

为了保证 ETag 头可用,我们必须使用条件请求。我们需要设置的第一个条件是 If-None-Match 头,这是一个用于 GET 的请求头。在后端接收到这个头之后,需要将它和当前的内容进行比较。如果值匹配的话,将只返回 304 状态码,和获取实体资源比较起来,响应结果的数据本身是很小的。这一切实施起来是非常简单的:如果你的第一个获取资源的 GET 请求返回了一个 ETag 数据,你的浏览器会自动的为接下来的资源的请求配置 If-None-Match 头。

这意味着,如果您的后端简单实现了 etagif-none-match,则可以减少从您的API传输到前端的数据量。

第二个请求条件使用的是 if-match 头。这被用来阻止 mid-air collisions 。 通俗的说,如果我们想要在后端更新数据,但是我们的前端数据已经过时,后端的更新应该被终止,而且前端也应该有提醒。 这和 if-none-match 的工作方式很类似。在获取到包含 ETag 值的资源之后,你可以提交一个 PATCH 请求并且设置一个和你之前接收到的 ETag 值相等的 If-Match 值。然后,后端将检查服务端当前可用的资源的 etag 值是否与您发送的资源相匹配。如果匹配,将允许您的更新。如果没有匹配,将返回412状态码,让前端知道条件不匹配。

如何使用

如果你想要在 laravel 项目中使用这个条件请求插件包的话,你可以使用以下命令来安装:

$ composer require werk365/etagconditionals

然后在你的路由中添加 etag 中间件之后就可以使用了。如果你想研究中间件的工作原理,或者想要不通过我们的插件包来实现这个功能的话,请接着往下读!

SetEtag 中间件

正如您可能已经猜到的那样,我们可以通过中间件来很简单的实现这一功能。 Laravel 实际上已经为我们提供了一个 SetCacheHeaders 中间件来设置 ETag 头, 但是它不支持 HEAD 请求。 SetEtag 中间件的内容看起来是这样的:

    public function handle(Request $request, Closure $next)
    {
        // Handle request
        $method = $request->getMethod();

        // Support using HEAD method for checking If-None-Match
        if ($request->isMethod('HEAD')) {
            $request->setMethod('GET');
        }

        //Handle response
        $response = $next($request);

        // Setting etag
        $etag = md5($response->getContent());
        $response->setEtag($etag);

        $request->setMethod($method);

        return $response;
    }

我们首先要做的事情是获取请求的方法,以防我们想要修改它。然后,当我们在处理 HEAD 请求的时候,我们把它修改为 GET 请求,以确保请求的内容已经被加载而且可以被加密。在此之后,我们跳转到已经使用 md5() 方法加密后的响应主体内容。在返回响应之前,我们会将加密后的 hash 值作为 ETag 头,并且将原始请求方法设置回原处。

IfNoneMatch 中间件

这是另一个相对简单的方法。让我们先来看看代码:

    public function handle(Request $request, Closure $next)
    {
        // Handle request
        $method = $request->getMethod();

        // Support using HEAD method for checking If-None-Match
        if ($request->isMethod('HEAD')) {
            $request->setMethod('GET');
        }

        //Handle response
        $response = $next($request);

        $etag = '"'.md5($response->getContent()).'"';
        $noneMatch = $request->getETags();

        if (in_array($etag, $noneMatch)) {
            $response->setNotModified();
        }

        $request->setMethod($method);

        return $response;
    }

这个开头与  SetEtag 中间件相似,将确保我们可以再次处理 HEAD 请求,并根据响应内容生成 hash值。注意这种情况下我们需要将hash值用双引号包裹。双引号包裹 ETag头,然后在setEtag中间件中 setEtag()方法自动包裹hash。有了hash值后,我们可以轻松的与 If-None-Match头进行比较。由于该头可以自动加载无限个hash,并且 getETags()方法会将它们以数组形式返回,所以我们可以核对新生成的值是否存在于数组中。如果确实有匹配,我们可以在响应中使用 setNotModified()设置 304 的状态码。

IfMatch 中间件

处理 If-Match 将稍微复杂一些。这个问题归结于:我们需要找到一种方法,用来获取应该更新的当前版本的内容。这可以用多种方式实现。

  • 你可以使用 HTTP 客户端对相同资源从外部发起 GET 请求
  • 你可以查看当前请求将执行的操作, 而不是调用与之等价的 GET 请求( 例如,调用控制器上的 show() 方法)
  • 或者你可以通过内部发起一个新的 GET 请求。

在构建这个中间件时,我开始尝试使用第二个选项。出于某种原因,这对我来说似乎是最好的选择。我成功地创建了一个完全可以工作的版本,但我对结果并不满意。为了让它工作,我需要做一些假设,预设一些限制,并做了太多的工作,而我只需要创建一个新的请求就可以了。

当我们要发起一个新的请求来获取当前版本资源的时候,代码是下面这样的:

    public function handle(Request $request, Closure $next)
    {
        // 只有请求方式是 PATCH 并且已经设置了 If-Match 头
        if (! ($request->isMethod('PATCH') && $request->hasHeader('If-Match'))) {
            return $next($request);
        }

        // 对同一个点创建新的 GET 请求,
        // 复制和添加请求头,让中间件能忽略本次请求
        $getRequest = Request::create($request->getRequestUri(), 'GET');
        $getRequest->headers = $request->headers;
        $getRequest->headers->set('X-From-Middleware', 'IfMatch');
        $getResponse = app()->handle($getRequest);

        // Get content from response object and get hashes from content and etag
        $getContent = $getResponse->getContent();
        $getEtag = '"'.md5($getContent).'"';
        $ifMatch = $request->header('If-Match');

        // 比较当前和请求携带的 hash 值
        if ($getEtag !== $ifMatch) {
            return response(null, 412);
        }

        return $next($request);

所有这些中间件都将在请求生命周期开始时运行。首先,我们将过滤掉任何非 PATCH 请求或请求头中没有 If Match 的请求。之后,我们将向同一个端点发出一个新的 GET 请求,并从原来的请求中复制请求头,以便新请求可以通过身份验证中间件和其他约束。

使用这个新请求的响应,我们将再次生成一个哈希,以便与发送的哈希进行比较。如果哈希匹配,请求将被中间件允许通过。如果不匹配,将返回状态代码为 412 的请求响应。

通过使用这三个中间件,你可以在你的 Laravel 应用程序中轻松处理 ETag 和条件请求。

软件包:https://github.com/365Werk/etagconditionals

原文地址:https://hergen.nl/caching-your-laravel-api-with-etag-and-conditional-requests

译文地址:https://learnku.com/laravel/t/55539

以上是详解如何使用ETag和条件标头进行缓存的详细内容。更多信息请关注PHP中文网其他相关文章!

声明
本文转载于:learnku。如有侵权,请联系admin@php.cn删除
使用Laravel:使用PHP简化Web开发使用Laravel:使用PHP简化Web开发Apr 19, 2025 am 12:18 AM

Laravel优化Web开发流程的方法包括:1.使用路由系统管理URL结构;2.利用Blade模板引擎简化视图开发;3.通过队列处理耗时任务;4.使用EloquentORM简化数据库操作;5.遵循最佳实践提高代码质量和可维护性。

Laravel:PHP Web框架的简介Laravel:PHP Web框架的简介Apr 19, 2025 am 12:15 AM

Laravel是一个现代化的PHP框架,提供了强大的工具集,简化了开发流程并提高了代码的可维护性和可扩展性。1)EloquentORM简化数据库操作;2)Blade模板引擎使前端开发直观;3)Artisan命令行工具提升开发效率;4)性能优化包括使用EagerLoading、缓存机制、遵循MVC架构、队列处理和编写测试用例。

Laravel:MVC建筑和最佳实践Laravel:MVC建筑和最佳实践Apr 19, 2025 am 12:13 AM

Laravel的MVC架构通过模型、视图、控制器分离数据逻辑、展示和业务处理,提高了代码的结构化和可维护性。1)模型处理数据,2)视图负责展示,3)控制器处理用户输入和业务逻辑,这种架构让开发者专注于业务逻辑,避免陷入代码泥潭。

Laravel:解释的主要功能和优势Laravel:解释的主要功能和优势Apr 19, 2025 am 12:12 AM

Laravel是一个基于MVC架构的PHP框架,具有简洁的语法、强大的命令行工具、便捷的数据操作和灵活的模板引擎。1.优雅的语法和易用的API使开发快速上手。2.Artisan命令行工具简化了代码生成和数据库管理。3.EloquentORM让数据操作直观简单。4.Blade模板引擎支持高级视图逻辑。

用Laravel建造后端:指南用Laravel建造后端:指南Apr 19, 2025 am 12:02 AM

Laravel适合构建后端服务,因为它提供了优雅的语法、丰富的功能和强大的社区支持。1)Laravel基于MVC架构,简化了开发流程。2)它包含EloquentORM,优化了数据库操作。3)Laravel的生态系统提供了如Artisan、Blade和路由系统等工具,提升开发效率。

laravel框架技巧分享laravel框架技巧分享Apr 18, 2025 pm 01:12 PM

在这个技术不断进步的时代,掌握先进的框架对于现代程序员至关重要。本文将通过分享 Laravel 框架中鲜为人知的技巧,帮助你提升开发技能。Laravel 以其优雅的语法和广泛的功能而闻名,本文将深入探讨其强大的特性,提供实用技巧和窍门,帮助你打造高效且维护性高的 Web 应用程序。

laravel和thinkphp的区别laravel和thinkphp的区别Apr 18, 2025 pm 01:09 PM

Laravel 和 ThinkPHP 都是流行的 PHP 框架,在开发中各有优缺点。本文将深入比较这两者,重点介绍它们的架构、特性和性能差异,以帮助开发者根据其特定项目需求做出明智的选择。

laravel用户登录功能一览laravel用户登录功能一览Apr 18, 2025 pm 01:06 PM

在 Laravel 中构建用户登录功能是一个至关重要的任务,本文将提供一个全面的概述,涵盖从用户注册到登录验证的每个关键步骤。我们将深入探讨 Laravel 的内置验证功能的强大功能,并指导您自定义和扩展登录过程以满足特定需求。通过遵循这些一步一步的说明,您可以创建安全可靠的登录系统,为您的 Laravel 应用程序的用户提供无缝的访问体验。

See all articles

热AI工具

Undresser.AI Undress

Undresser.AI Undress

人工智能驱动的应用程序,用于创建逼真的裸体照片

AI Clothes Remover

AI Clothes Remover

用于从照片中去除衣服的在线人工智能工具。

Undress AI Tool

Undress AI Tool

免费脱衣服图片

Clothoff.io

Clothoff.io

AI脱衣机

Video Face Swap

Video Face Swap

使用我们完全免费的人工智能换脸工具轻松在任何视频中换脸!

热工具

EditPlus 中文破解版

EditPlus 中文破解版

体积小,语法高亮,不支持代码提示功能

PhpStorm Mac 版本

PhpStorm Mac 版本

最新(2018.2.1 )专业的PHP集成开发工具

禅工作室 13.0.1

禅工作室 13.0.1

功能强大的PHP集成开发环境

WebStorm Mac版

WebStorm Mac版

好用的JavaScript开发工具

螳螂BT

螳螂BT

Mantis是一个易于部署的基于Web的缺陷跟踪工具,用于帮助产品缺陷跟踪。它需要PHP、MySQL和一个Web服务器。请查看我们的演示和托管服务。