0

0

Laravel Eloquent:同时筛选父子表数据的教程

碧海醫心

碧海醫心

发布时间:2025-11-30 11:59:01

|

693人浏览过

|

来源于php中文网

原创

Laravel Eloquent:同时筛选父子表数据的教程

本教程详细阐述了如何在 laravel 中利用 eloquent orm 同时对父表和子表数据进行筛选。文章将深入探讨两种核心方法:使用 `join` 子句进行直接数据库连接,以及采用 `wherehas` 方法实现更具 eloquent 风格的关联查询。通过实际代码示例,您将学会如何根据父表的字段(如年份)和子表的字段(如标签id)构建高效且可读的过滤逻辑,并集成到控制器和视图中。

在 Laravel 应用开发中,我们经常需要处理具有一对多或多对多关系的数据。当用户需要根据父表和其关联子表的字段同时进行筛选时,如何高效地构建查询是关键。本教程将以 Post (父表) 和 PostTag (子表) 为例,演示如何实现这一需求,具体涉及根据 Post 表的 year 字段和 PostTag 表的 tag_id 字段进行过滤。

1. 模型及关系定义

首先,确保您的 Eloquent 模型已正确定义了它们之间的关系。在本例中,Post 模型与 PostTag 模型之间存在一对多关系。

Post.php

hasMany(PostTag::class);
    }
}

PostTag.php

belongsTo(Post::class);
    }
}

2. 视图中的筛选表单

用户通过一个包含年份和标签选择的表单来提交筛选请求。

View (movies/search_form.blade.php)

3. 路由定义

定义一个 GET 路由来处理筛选请求。

web.php

Route::prefix('/movies')->group(function () {
    Route::get('/search', [MovieController::class, 'MoviesDataSearch'])->name('movies.movie_search');
});

4. 控制器中的筛选逻辑

在控制器中,我们将介绍两种实现父子表同时筛选的方法:使用 join 子句和使用 whereHas 方法。

4.1 方法一:使用 join 子句

join 子句允许您直接在数据库层面连接两个或多个表,然后对连接后的结果进行筛选。这种方法通常在需要高性能或需要直接访问连接表中所有列时非常有用。

优点:

LobeHub
LobeHub

LobeChat brings you the best user experience of ChatGPT, OLLaMA, Gemini, Claude

下载
  • 性能通常较好,尤其是在处理大量数据时。
  • 可以直接在查询中访问所有连接表的列。

缺点:

  • 如果不对 select 语句进行处理,可能会出现列名冲突。
  • 如果一个父记录有多个匹配的子记录,可能会导致父记录重复出现,需要使用 distinct() 或 groupBy() 来避免。

MovieController.php (使用 join)

where('posts.post_type', "movie")
              ->where('posts.is_delete', "0");

        // 如果存在标签筛选,则使用 join 连接 posts_tags 表
        if ($request->filled('tag')) {
            // 注意:'posts_tags.post_id' 是 PostTag 模型中指向 Post 模型的外键
            $query->join('posts_tags', 'posts.id', '=', 'posts_tags.post_id')
                  ->where('posts_tags.tag_id', $request->tag); // 假设 tag_id 是精确匹配
            // 如果需要模糊匹配,可以使用 ->where('posts_tags.tag_id', 'like', '%'.$request->tag.'%');
        }

        // 如果存在年份筛选
        if ($request->filled('year')) {
            $query->where('posts.year', 'like', '%'.$request->year.'%');
        }

        // 避免因 join 导致重复的 Post 记录,并仅选择 Post 表的列
        $movies = $query->select('posts.*')
                        ->distinct() // 确保每个 Post 记录只出现一次
                        ->orderBy('posts.id', 'DESC')
                        ->paginate(12); // 分页大小可根据需求调整

        // 保持筛选参数在分页链接中
        return $movies->appends($request->all());
    }
}

注意事项:

  • join('posts_tags', 'posts.id', '=', 'posts_tags.post_id'): 这里的 posts_tags.post_id 是 PostTag 表中关联 Post 表的外键。请根据您的实际数据库结构调整。
  • select('posts.*'): 在使用 join 时,为了避免返回重复的列名或不必要的子表数据,通常建议明确指定要选择的列。
  • distinct(): 如果一个 Post 记录有多个 PostTag 匹配筛选条件,join 可能会导致该 Post 记录在结果集中出现多次。distinct() 可以确保每个 Post 记录只返回一次。

4.2 方法二:使用 whereHas 方法

whereHas 是 Eloquent 提供的一种更高级、更具对象化风格的方法,用于根据关联模型的条件来筛选父模型。它会在底层生成一个 EXISTS 子查询。

优点:

  • 代码更简洁,更符合 Eloquent 的设计理念。
  • 自动处理连接逻辑,无需手动指定 join 条件。
  • 避免了 join 可能导致的重复父记录问题。

缺点:

  • 对于非常复杂的关联查询或需要高度优化性能的场景,有时 join 可能更直接高效。
  • 无法直接选择关联模型的列。

MovieController.php (使用 whereHas)

where('post_type', "movie")
              ->where('is_delete', "0");

        // 如果存在年份筛选
        if ($request->filled('year')) {
            $query->where('year', 'like', '%'.$request->year.'%');
        }

        // 如果存在标签筛选,则使用 whereHas
        if ($request->filled('tag')) {
            $query->whereHas('PostTags', function ($q) use ($request) {
                // 在关联模型 PostTag 上应用筛选条件
                $q->where('tag_id', $request->tag); // 假设 tag_id 是精确匹配
                // 如果需要模糊匹配,可以使用 $q->where('tag_id', 'like', '%'.$request->tag.'%');
            });
        }

        $movies = $query->orderBy('id', 'DESC')
                        ->paginate(12);

        // 保持筛选参数在分页链接中
        return $movies->appends($request->all());
    }
}

注意事项:

  • whereHas('PostTags', function ($q) use ($request) { ... }): PostTags 是在 Post 模型中定义的关系方法名。闭包函数 $q 代表了对 PostTag 模型进行的查询。
  • $request->filled('year') 和 $request->filled('tag'): 使用 filled() 方法可以检查请求参数是否存在且非空,这有助于构建更健壮的条件查询。

5. 总结与最佳实践

  • 选择方法:
    • 如果您需要直接从关联表中选择列,或者对性能有极高要求且熟悉 SQL JOIN 语句的优化,join 方法可能更适合。
    • 如果您追求代码的 Eloquent 风格、可读性,并且主要目的是根据关联条件筛选父模型,那么 whereHas 是更推荐的选择。它通常能满足大部分业务需求,并且更不容易引入重复记录的问题。
  • 条件查询: 在控制器中,使用 if ($request->filled('param')) 或 Eloquent 的 when() 方法来根据请求参数的有无动态构建查询条件,可以使代码更加灵活和健壮。
    // 使用 when() 的示例
    $movies = Post::where('post_type', "movie")
        ->where('is_delete', "0")
        ->when($request->filled('year'), function ($query) use ($request) {
            return $query->where('year', 'like', '%'.$request->year.'%');
        })
        ->when($request->filled('tag'), function ($query) use ($request) {
            return $query->whereHas('PostTags', function ($q) use ($request) {
                $q->where('tag_id', $request->tag);
            });
        })
        ->orderBy('id', 'DESC')
        ->paginate(12);
  • 分页链接: 务必使用 $movies->appends($request->all()) 来确保在分页时,所有当前的筛选参数都能被保留,从而提供更好的用户体验。
  • 外键匹配: 无论是 join 还是 whereHas,都依赖于正确的外键关系。请确保您的数据库表和 Eloquent 模型中的外键定义是准确无误的。
  • 输入验证: 在实际应用中,始终对用户输入进行验证(例如,使用 Laravel 的表单请求验证),以防止恶意数据或不符合预期的数据影响查询。

通过掌握 join 和 whereHas 这两种强大的 Eloquent 查询方法,您将能够灵活高效地处理 Laravel 中父子表数据的复杂筛选需求。

相关专题

更多
php文件怎么打开
php文件怎么打开

打开php文件步骤:1、选择文本编辑器;2、在选择的文本编辑器中,创建一个新的文件,并将其保存为.php文件;3、在创建的PHP文件中,编写PHP代码;4、要在本地计算机上运行PHP文件,需要设置一个服务器环境;5、安装服务器环境后,需要将PHP文件放入服务器目录中;6、一旦将PHP文件放入服务器目录中,就可以通过浏览器来运行它。

2644

2023.09.01

php怎么取出数组的前几个元素
php怎么取出数组的前几个元素

取出php数组的前几个元素的方法有使用array_slice()函数、使用array_splice()函数、使用循环遍历、使用array_slice()函数和array_values()函数等。本专题为大家提供php数组相关的文章、下载、课程内容,供大家免费下载体验。

1656

2023.10.11

php反序列化失败怎么办
php反序列化失败怎么办

php反序列化失败的解决办法检查序列化数据。检查类定义、检查错误日志、更新PHP版本和应用安全措施等。本专题为大家提供php反序列化相关的文章、下载、课程内容,供大家免费下载体验。

1513

2023.10.11

php怎么连接mssql数据库
php怎么连接mssql数据库

连接方法:1、通过mssql_系列函数;2、通过sqlsrv_系列函数;3、通过odbc方式连接;4、通过PDO方式;5、通过COM方式连接。想了解php怎么连接mssql数据库的详细内容,可以访问下面的文章。

952

2023.10.23

php连接mssql数据库的方法
php连接mssql数据库的方法

php连接mssql数据库的方法有使用PHP的MSSQL扩展、使用PDO等。想了解更多php连接mssql数据库相关内容,可以阅读本专题下面的文章。

1418

2023.10.23

html怎么上传
html怎么上传

html通过使用HTML表单、JavaScript和PHP上传。更多关于html的问题详细请看本专题下面的文章。php中文网欢迎大家前来学习。

1234

2023.11.03

PHP出现乱码怎么解决
PHP出现乱码怎么解决

PHP出现乱码可以通过修改PHP文件头部的字符编码设置、检查PHP文件的编码格式、检查数据库连接设置和检查HTML页面的字符编码设置来解决。更多关于php乱码的问题详情请看本专题下面的文章。php中文网欢迎大家前来学习。

1448

2023.11.09

php文件怎么在手机上打开
php文件怎么在手机上打开

php文件在手机上打开需要在手机上搭建一个能够运行php的服务器环境,并将php文件上传到服务器上。再在手机上的浏览器中输入服务器的IP地址或域名,加上php文件的路径,即可打开php文件并查看其内容。更多关于php相关问题,详情请看本专题下面的文章。php中文网欢迎大家前来学习。

1306

2023.11.13

高德地图升级方法汇总
高德地图升级方法汇总

本专题整合了高德地图升级相关教程,阅读专题下面的文章了解更多详细内容。

68

2026.01.16

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
PHP课程
PHP课程

共137课时 | 8.8万人学习

JavaScript ES5基础线上课程教学
JavaScript ES5基础线上课程教学

共6课时 | 7.9万人学习

PHP新手语法线上课程教学
PHP新手语法线上课程教学

共13课时 | 0.9万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号