Laravel渴求式加载(比较容易理解理解load与with关系)

Laravel渴求式加载(比较容易理解理解load与with关系)

渴求式加载

当以属性方式访问 Eloquent关联关系的时候,关联关系数据是「懒惰式加载」的,这意味着关联关系数据直到第一次访问的时候才被加载。不过,Eloquent 还可以在查询父级模型的同时「渴求式加载」关联关系。渴求式加载缓解 N+1 查询问题,要阐明 N+1 查询问题,查看关联到 Auth的 Book模型:

Laravel学院文档: https://laravelacademy.org/post/9584.html

根据官方文档, 创建两个表,不多说.
Laravel渴求式加载(比较容易理解理解load与with关系)
新建两个model模型

php artisan make:model Auth
php artisan mkae:model Book

然后创建对应的表关联关系

<?php
namespace App;

use Illuminate\Database\Eloquent\Model;

class Book extends Model
{
    protected  $table = 'book';

	
    public function auth()
    {
        return $this->belongsTo('App\Auth');
    }
}

在控制器中获取书的作者信息

  public function lst()
    {
        $books  = Book::all();

        foreach ($books as $book){
        echo $book->auth->name;
        }
    }

去页面查看输出信息和SQL语句
Laravel渴求式加载(比较容易理解理解load与with关系)

这里发现 其实这里是执行了5条sql语句,在foreach循环中,每一次都执行了一条sql语句

文档中的解释:

该循环先执行 1 次查询获取表中的所有书,然后另一个查询获取每一本书的作者,因此,如果有25本书,要执行26次查询:1次是获取书本身,剩下的25次查询是为每一本书获取其作者。
在这里 我们使用渴求式加载. 当查询的时候,可以使用 with方法指定应该被渴求式加载的关联关系:

 public function lst()
    {
        $books  = Book::with('auth')->get();

        foreach ($books as $book){
        echo $book->auth->name;
        }
    }

查看输出结果,发现,效果一样,但原本执行的多条sql 变成了两条
Laravel渴求式加载(比较容易理解理解load与with关系)

Laravel渴求式加载(比较容易理解理解load与with关系)

渴求式加载多个关联关系

有时候你需要在单个操作中渴求式加载多个不同的关联关系。要实现这个功能,只需要添加额外的参数到with方法即可:

在这里,我又新建了一个category表
Laravel渴求式加载(比较容易理解理解load与with关系)
在Book模型中又加了一个与category表的关联关系

public function category()
    {
        //这里我在book表中与category表关联的外键字段是cate_id,而laravel中默认是category_id,所以修改第二个参数为cate_id
        return $this->belongsTo('App\Category','cate_id');
    }

修改代码

 public function lst()
    {
        $books  = Book::with('auth','category')->get();

        foreach ($books as $book){
            echo $book->category->name;
            echo $book->auth->name;
        }
    }

查看页面输出结果,这里将category表中的数据也查出来了
Laravel渴求式加载(比较容易理解理解load与with关系)
嵌套的渴求式加载

要渴求式加载嵌套的关联关系,可以使用”.“语法。例如,我们在一个 Eloquent 语句中渴求式加载所有书的作者及所有作者的个人联系方式:
$books = Book::with(‘auth.contact’)->get();

这里还需要新建一个contact表,存放作者的联系方式
Laravel渴求式加载(比较容易理解理解load与with关系)
然后新建一个Contact模型,并在Auth模型中关联上Contact模型

<?php

namespace App;

use Illuminate\Database\Eloquent\Model;

class Auth extends Model
{
    protected $table = 'auth';

    public function contact()
    {
        return $this->belongsTo('App\Contact');
    }
}

然后修改控制器中的代码

public function lst()
    {
        //通过 . 点号 来加载auth模型的contact关联表
        $books  = Book::with('auth.contact')->get();
        
        echo '<pre>';
        
        foreach ($books as $book){
            echo $book->name;
            echo '<br/>';
            echo $book->auth->name;
            echo '<br/>';
            echo $book->auth->contact->email;
            echo '<hr/>';
        }
    }

查看页面输出信息,这里将auth关联的contact表的信息也查出来了.
Laravel渴求式加载(比较容易理解理解load与with关系)
渴求式加载指定字段

并不是每次获取关联关系时都需要所有字段,因此,Eloquent 允许你在关联查询时指定要查询的字段:

$users = Book::with('auth:id,name')->get();   

注:使用这个特性时,id字段是必须列出的。

public function lst()
    {
        $books  = Book::with('auth:id,name')->get()->toArray();
        dd($books);
    }

查看页面输出代码 auth下只有id和name两个字段了 其他多余的字段都没有了

Laravel渴求式加载(比较容易理解理解load与with关系)
带条件约束的渴求式加载

有时候我们希望渴求式加载一个关联关系,但还想为渴求式加载指定更多的查询条件:

$books = Book::with(['auth' => function ($query) {
    //约束条件
    $query->where('name', 'like', '吴%');
}])->get();
dd($books);

查看页面输出信息,只有auth表中name字段以吴开头的信息查出来了,其他的都没查出来
Laravel渴求式加载(比较容易理解理解load与with关系)
在这个例子中,Eloquent 只渴求式加载 name 以吴开头 的作者。当然,你还可以调用其它查询构建器来自定义渴求式加载操作:

$books = Book::with(['auth' => function ($query) {
    $query->orderBy('name', 'desc');
}])->get();

懒惰渴求式加载

有时候你需要在父模型已经被获取后渴求式加载一个关联关系。例如,这在你需要动态决定是否加载关联模型时可能很有用:

$books  =  Book::all();
        $res = true;
        if($res){
            $books->load('auth','category');
        }
        dd(collect($books)->toArray());

如果你需要设置更多的查询条件到渴求式加载查询上,可以传递一个包含你想要记载的关联关系数组到 load 方法,数组的值应该是接收查询实例的闭包:

$books->load(['auth' => function ($query) {
    $query->orderBy('name', 'asc');
}]);

如果想要在关系管理尚未被加载的情况下加载它,可以使用 loadMissing 方法:

public function lst(Book $book)
{
    $book->loadMissing('auth');

    return [
        'name' => $book->name,
        'auth' => $book->auth->name
    ];
}

最后这个…试了很久,在大佬的帮助下完成了.
直接从文档复制过来就报错了!!!坑啊
需要在路由那传递一个id

Route::get('book/{id}','ArticleController@lst');

Laravel渴求式加载(比较容易理解理解load与with关系)
去掉loadMissing之后…下面那行sql语句变了,但是感觉没啥用啊.不知道这个东西有什么用.
Laravel渴求式加载(比较容易理解理解load与with关系)Laravel渴求式加载(比较容易理解理解load与with关系)

 

原文:https://blog.csdn.net/u013032345/article/details/82772938

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 举报,一经查实,本站将立刻删除。

发布者:全栈程序员-用户IM,转载请注明出处:https://javaforall.cn/112397.html原文链接:https://javaforall.cn

【正版授权,激活自己账号】: Jetbrains全家桶Ide使用,1年售后保障,每天仅需1毛

【官方授权 正版激活】: 官方授权 正版激活 支持Jetbrains家族下所有IDE 使用个人JB账号...

(0)


相关推荐

  • 浅谈辄止_java forkjoinpool

    浅谈辄止_java forkjoinpool文章目录一、ForkJoin是什么?它能用来实现什么功能?二、ForkJoin的实现原理三、ForkJoin的简单使用一、ForkJoin是什么?它能用来实现什么功能?二、ForkJoin的实现原理三、ForkJoin的简单使用在这里插入代码片…

  • java反射菜鸟教程_Java反射

    java反射菜鸟教程_Java反射JAVA反射机制1定义:主要是指程序可以访问,检测和修改它本身状态或行为的一种能力,并能根据自身行为的状态和结果,调整或修改应用所描述行为的状态和相关的语义。反射是java中一种强大的工具,能够使我们很方便的创建灵活的代码,这些代码可以再运行时装配,无需在组件之间进行源代码链接。但是反射使用不当会成本很高!2.作用A:反编译:.class–>.javaB:通过反射机制访问java对象的属性…

  • PAT乙级1004

    PAT乙级10041004.成绩排名(20)时间限制400ms内存限制65536kB代码长度限制8000B判题程序Standard作者CHEN,Yue读入n名学生的姓名、学号、成绩,分别输出成绩最高和成绩最低学生的姓名和学号。输入格式:每个测试输入包含1个测试用例,格式为第1行:正整数n第2行:第1个学生的姓名学号成绩第3行:第2个学生的姓名学号成绩…………

  • java中Switch(expr)里面的参数的类型是?

    java中Switch(expr)里面的参数的类型是?在Java5以前,switch(expr)中,exper只能是byte,short,char,int类型在JDK1.6:引入enumJDK1.7时:又增加了Stringpublic static void main(String[] args) { switch ("123") { case "123": System.out.println("1234"); b…

  • C语言——冒泡排序[通俗易懂]

    C语言——冒泡排序[通俗易懂]冒泡排序的原理是:从左到右,相邻元素进行比较。每次比较一轮,就会找到序列中最大的一个或最小的一个。这个数就会从序列的最右边冒出来。以从小到大排序为例,第一轮比较后,所有数中最大的那个数就会浮到最右边;第二轮比较后,所有数中第二大的那个数就会浮到倒数第二个位置……就这样一轮一轮地比较,最后实现从小到大排序。代码如下#include<stdio.h>voidBBsort(intarr[],intsize){ intj,i,tem; for(i=0;i..

    2022年10月23日
  • 企业微信通讯录回调密文解析及微信支付回调密文解析异常处理

    企业微信通讯录回调密文解析及微信支付回调密文解析异常处理企业微信通讯录回调密文解析及微信支付回调密文解析异常处理产生异常表现:javax.crypto.IllegalBlockSizeException:Inputlengthmustbemultipleof16whendecryp原因:因为某些国家的进口管制限制,Java发布的运行环境包中的加解密有一定的限制解决方案:替换jdk和jre中的local_policy.jar,US_export_policy.jar具体目录:1、jre目录/lib/security/policy/

发表回复

您的电子邮箱地址不会被公开。

关注全栈程序员社区公众号