千家信息网

Laravel8如何通过禁用延迟来定位N+1问题

发表于:2024-10-17 作者:千家信息网编辑
千家信息网最后更新 2024年10月17日,这篇文章给大家分享的是有关Laravel8如何通过禁用延迟来定位N+1问题的内容。小编觉得挺实用的,因此分享给大家做个参考,一起跟随小编过来看看吧。在Laravel 8的下一个版本中,您可以完全禁用延
千家信息网最后更新 2024年10月17日Laravel8如何通过禁用延迟来定位N+1问题

这篇文章给大家分享的是有关Laravel8如何通过禁用延迟来定位N+1问题的内容。小编觉得挺实用的,因此分享给大家做个参考,一起跟随小编过来看看吧。

在Laravel 8的下一个版本中,您可以完全禁用延迟加载,从而导致异常:

防止N+1问题? @themsaid最近对框架的贡献允许您完全禁用延迟加载(将引发异常)...

只能在非生产环境下禁用它,这样在一个进程中出错时生产不会崩溃!

下周发布! pic.twitter.com/5Q9YpCLRze

- Taylor Otwell (@taylorotwell) May 19, 2021

防止开发中的延迟加载可以帮助您在开发过程的早期捕获N+1错误。Laravel生态系统有各种工具来识别N+1查询。然而,这种方法通过抛出一个异常来将问题放在前面和中心。

演示

让我们快速浏览一下这个特性,通过旋转框架8.x分支的开发版本,因为在撰写本文时这个特性还没有推出。一旦发布,您将拥有此功能,而无需切换到最新的8.x分支。

安装

首先,创建一个新的应用程序:

laravel new strict-lazy-demo

接下来,我们将更新composer.json中的laravel/framework版本,通过将版本调整为8.x-dev,确保我们拥有此功能(如果您在下一版本之前尝试此功能):

{    "require": {        "laravel/framework": "8.x-dev"    }}

接下来,运行composer update以确保获得此分支的最新版本代码:

composer update laravel/framework

此时,您应该设置首选数据库。我喜欢使用Laravel的默认值运行本地MySQL实例,即使用root用户而不使用密码。我发现在本地使用默认的.env值很方便,无需任何配置即可快速开始。

mysql -uroot -e"create database strict_lazy_demo"

配置所选数据库后,请确保可以迁移:

php artisan migrate:fresh

Demo Data

我们将创建一个Post模型,并从User模型中定义一对多关系,以演示此功能。我们将首先创建Post模型和附带的文件:

# 使用迁移和工厂创建模型php artisan make:model -mf Post

首先,让我们定义Post迁移和工厂配置:

// 您的文件名将根据创建文件的时间而有所不同。// 2021_05_21_000013_create_posts_table.phpSchema::create('posts', function (Blueprint $table) {    $table->id();    $table->foreignIdFor(\App\Models\User::class);    $table->string('title');    $table->longText('body');    $table->timestamps();});

接下来,根据上述模式定义PostFactory定义方法:

/** * Define the model's default state. * * @return array */public function definition(){    return [        'user_id' => \App\Models\User::factory(),        'title' => $this->faker->sentence(),        'body' => implode("\n\n", $this->faker->paragraphs(rand(2,5))),    ];}

最后,打开DatabaseSeeder文件,并在run()方法中添加以下内容:

/** * 数据库填充程序。 * * @return void */public function run(){    \App\Models\User::factory()        ->has(\App\Models\Post::factory()->count(3))        ->create()    ;}

关联模型并防止延迟加载

现在我们已经创建了迁移文件、填充文件和模型,我们已经准备好将User与Post模型关联起来以演示该特性。向User模型添加以下方法,以给用户一个与Posts的关联:

// app/Models/User.php/** * @return \Illuminate\Database\Eloquent\Relations\HasMany */public function posts(){    return $this->hasMany(Post::class);}

有了这些,我们就可以迁移和填充数据库了:

php artisan migrate:fresh --seed

如果一切顺利,我们将在控制台中看到如下内容:

现在,我们可以使用tinker检查我们的填充数据和关系:

php artisan tinker>>> $user = User::first()=> App\Models\User {#4091     id: 1,     name: "Nedra Hayes",     email: "bruen.marc@example.com",     email_verified_at: "2021-05-21 00:35:59",     created_at: "2021-05-21 00:35:59",     updated_at: "2021-05-21 00:35:59",   }>>> $user->posts=> Illuminate\Database\Eloquent\Collection {#3686     all: [       App\Models\Post {#3369         id: 1,...

$user->posts 属性实际上调用了数据库,因此是"惰性"的,但没有进行优化。延迟加载的便利性很好,但从长远来看,它可能带来沉重的性能负担。

禁用延迟加载

现在我们已经设置了模型,我们可以在应用程序中禁用延迟加载。您可能只希望在非生产环境中禁用,这很容易实现!打开"AppServiceProvider"类并将以下内容添加到"boot()"方法:

// app/Providers/AppServiceProvider.phppublic function boot(){    Model::preventLazyLoading(! app()->isProduction());}

当你再次运行 php artisan tinker , 此时您应该会收到延迟加载违规的异常:

php artisan tinker>>> $user = \App\Models\User::first()=> App\Models\User {#3685     id: 1,     name: "Nedra Hayes",     email: "bruen.marc@example.com",     email_verified_at: "2021-05-21 00:35:59",     #password: "$2y$10$92IXUNpkjO0rOQ5byMi.Ye4oKoEa3Ro9llC/.og/at2.uheWG/igi",     #remember_token: "jHSxFGKOdw",     created_at: "2021-05-21 00:35:59",     updated_at: "2021-05-21 00:35:59",   }>>> $user->postsIlluminate\Database\LazyLoadingViolationException with message'Attempted to lazy load [posts] on model [App\Models\User] but lazy loading is disabled.'

如果要观察在视图中使用延迟加载时发生的情况,请按照如下方式修改默认路由:

Route::get('/', function () {    return view('welcome', [        'user' => \App\Models\User::first()    ]);});

接下来,在 welcome.blade.php 文件中某处添加以下内容:

Posts

@foreach($user->posts as $post)

{{ $post->title }}

{{ $post->body }}

@endforeach

如果您通过 Valet 或 artisan serve 加载您的应用程序,您应该出现异常。

尽管您在开发过程中会遇到异常,但只要您在服务提供者中正确设置了环境检查,意外部署触发延迟加载的代码将继续工作。

感谢各位的阅读!关于"Laravel8如何通过禁用延迟来定位N+1问题"这篇文章就分享到这里了,希望以上内容可以对大家有一定的帮助,让大家可以学到更多知识,如果觉得文章不错,可以把它分享出去让更多的人看到吧!

延迟 模型 文件 内容 数据 版本 数据库 方法 问题 接下来 功能 程序 开发 分支 应用程序 特性 环境 关联 应用 演示 数据库的安全要保护哪些东西 数据库安全各自的含义是什么 生产安全数据库录入 数据库的安全性及管理 数据库安全策略包含哪些 海淀数据库安全审计系统 建立农村房屋安全信息数据库 易用的数据库客户端支持安全管理 连接数据库失败ssl安全错误 数据库的锁怎样保障安全 互联网科技变革的根本原因 网络技术学习桌测评 浩少网络技术 仿英语教育app软件开发 宿州软件开发 建立物资采购数据库 传统服务器提供商可以消失吗 闽侯网络安全招投标 深泽县网络安全广告征集活动 工控系统国产化对网络安全意义 蒂森克虏伯服务器 电脑装服务器主板怎么样 mt管理器修改服务器验证 手机软件开发培训机构 安徽高科技软件开发创新服务 网络安全教育总结900字 软件开发开发和测试时间占比 计算机网络技术多少钱 登录小米账号时服务器错误怎么办 网络安全真实案列视频 钱海网络技术有限公司地址 访问数据库出错 sql 网络安全专业考研方向 软件开发京东自营 计算机网络技术实习做什么工作 邳州专业性软件开发常见问题 用友安装服务器通常选择那一个 通过路由器无线访问服务器 软件开发公司如何接活 网络安全产业分析报告
0