0

0

Laravel教程:使用 whereIn 实现多分类文章高效筛选

花韻仙語

花韻仙語

发布时间:2025-11-19 14:59:22

|

739人浏览过

|

来源于php中文网

原创

Laravel教程:使用 whereIn 实现多分类文章高效筛选

本教程详细介绍了在laravel中如何正确实现文章的多分类筛选功能。针对传统 `where` 循环导致筛选失败的问题,我们引入并演示了 `wherein` 方法的正确用法,它能高效处理多个分类条件的逻辑或(or)查询,确保用户选择多个分类时,系统能准确返回符合任一选中分类的文章,从而优化用户体验和查询性能。

理解多分类筛选的需求

在Web应用中,用户经常需要根据多个条件来筛选内容,例如根据多个标签、多个作者或多个分类来查找文章。对于文章分类筛选,常见的需求是:当用户选择“摄影”和“绘画”这两个分类时,系统应返回所有属于“摄影”分类属于“绘画”分类的文章。这意味着我们需要一个逻辑“或”(OR)的关系来组合这些筛选条件。

初始实现与遇到的问题

许多开发者在实现多分类筛选时,可能会尝试通过循环遍历用户提交的分类列表,并为每个分类添加一个 where 子句。以下是这种常见但存在问题的实现方式:

Blade 视图(前端表单示例):

@if (isset($categories))
  @foreach ($categories as $category)
    
@endforeach @endif

前端通过 name="cate[]" 将选中的分类 slug 作为数组发送到后端

控制器(问题代码示例):

public function search(Request $request)
{
    $categories = Category::all();
    $query = Post::orderBy('id', 'DESC'); // 基础查询

    // ... 其他筛选条件,例如关键词搜索 ...

    if ($request->has('cate')) {
        $categoryType = $request->input('cate'); // 获取选中的分类 slug 数组
        foreach ($categoryType as $category) {
            $query->where('category_slug', $category); // 为每个分类添加 where 子句
        }
    }

    $queryResults = $query->paginate(20);
    return view('searchPage', ['categories' => $categories, 'queryResults' => $queryResults]);
}

这段代码的问题在于,Laravel 的查询构建器在默认情况下,连续的 where 子句之间会使用逻辑“与”(AND)进行连接。这意味着如果用户选择了“摄影”和“绘画”两个分类,生成的 SQL 查询会类似于 WHERE category_slug = 'photography' AND category_slug = 'drawing'。一个文章不可能同时拥有两个不同的 category_slug,因此这样的查询将永远返回空结果。

解决方案:使用 whereIn 方法

为了解决上述问题,我们需要使用 whereIn 方法。whereIn 是 Laravel 查询构建器提供的一个强大功能,它允许你指定一个列,并检查该列的值是否存在于一个给定的数组中。这完美地符合了“逻辑或”(OR)的需求。

星火作家大神
星火作家大神

星火作家大神是一款面向作家的AI写作工具

下载

当用户选择多个分类时,例如 ['photography', 'drawing'],whereIn('category_slug', ['photography', 'drawing']) 将会生成类似于 WHERE category_slug IN ('photography', 'drawing') 的 SQL 查询,这正是我们期望的“或”逻辑。

修正后的控制器代码片段:

public function search(Request $request)
{
    $categories = Category::all();
    $query = Post::orderBy('id', 'DESC'); // 基础查询

    $txtSearch = $request->input('q');
    if (isset($txtSearch)) {
        $query->where('title', 'LIKE', "%$txtSearch%");
    }

    // 关键修正:使用 whereIn 处理多分类筛选
    if ($request->has('cate')) {
        // 确保 $request->input('cate') 始终是一个数组,即使没有选中任何项或输入格式不正确
        // 第二个参数 [] 是默认值,当 'cate' 不存在时返回空数组
        $selectedCategories = (array) $request->input('cate', []);
        if (!empty($selectedCategories)) { // 只有当有选中的分类时才应用 whereIn
            $query->whereIn('category_slug', $selectedCategories);
        }
    }

    $queryResults = $query->paginate(20);
    return view('searchPage', ['categories' => $categories, 'queryResults' => $queryResults]);
}

代码解析:

  1. $request->input('cate', []): 从请求中获取 cate 参数。如果 cate 不存在,则返回一个空数组 [],避免潜在的错误。
  2. (array) 类型转换:为了确保 whereIn 方法接收的第二个参数始终是一个数组,我们进行了显式的类型转换。虽然通常情况下,通过 name="cate[]" 提交的表单数据会自动解析为数组,但这是一个良好的编程习惯,可以增加代码的健壮性。
  3. if (!empty($selectedCategories)): 这是一个额外的安全检查。如果用户提交了 cate 参数,但实际上没有选中任何分类(例如,前端逻辑错误导致发送了一个空数组),此条件可以防止 whereIn 被调用,从而避免生成可能不必要的 WHERE column IN () SQL,提高查询效率。

完整的控制器示例

结合关键词搜索和多分类筛选的完整控制器代码如下:

input('q');
        if (!empty($txtSearch)) {
            $query->where('title', 'LIKE', "%{$txtSearch}%");
        }

        // 处理多分类筛选
        if ($request->has('cate')) {
            $selectedCategories = (array) $request->input('cate', []);
            if (!empty($selectedCategories)) {
                $query->whereIn('category_slug', $selectedCategories);
            }
        }

        // 执行查询并分页
        $queryResults = $query->paginate(20);

        // 返回视图
        return view('searchPage', [
            'categories' => $categories,
            'queryResults' => $queryResults
        ]);
    }
}

注意事项与最佳实践

  • 输入验证: 在实际生产环境中,强烈建议对用户输入进行验证。例如,确保 cate 参数确实是一个数组,并且数组中的每个值都是有效的分类 slug。Laravel 提供了强大的验证功能:
    $request->validate([
        'cate' => 'nullable|array',
        'cate.*' => 'exists:categories,slug', // 确保每个slug都存在于categories表的slug列中
    ]);
  • 性能考量: whereIn 在处理大量值时效率很高。然而,如果 whereIn 数组中的元素数量非常庞大(例如数千个),可能会对数据库性能产生影响。在这种极端情况下,可能需要考虑其他优化策略,例如使用临时表或更复杂的 JOIN 查询。但对于常规的多选筛选,whereIn 是一个高效且简洁的选择。
  • 可读性: whereIn 方法极大地提高了代码的可读性,清晰地表达了“字段值在给定集合中”的逻辑。
  • 错误处理: 始终考虑用户输入可能不符合预期的情况,例如空数组、非数组值等,并进行适当的处理,如本教程中使用的 (array) 强制转换和 empty() 检查。

总结

通过本教程,我们了解了在Laravel中实现多分类文章筛选的正确方法。摒弃了容易导致逻辑错误的 where 循环,转而采用 Laravel 查询构建器提供的 whereIn 方法,能够优雅且高效地处理多个条件之间的“或”关系。结合输入验证和健壮性处理,我们可以构建出功能完善、性能优良且易于维护的筛选功能。

相关专题

更多
laravel组件介绍
laravel组件介绍

laravel 提供了丰富的组件,包括身份验证、模板引擎、缓存、命令行工具、数据库交互、对象关系映射器、事件处理、文件操作、电子邮件发送、队列管理和数据验证。想了解更多laravel的相关内容,可以阅读本专题下面的文章。

319

2024.04.09

laravel中间件介绍
laravel中间件介绍

laravel 中间件分为五种类型:全局、路由、组、终止和自定。想了解更多laravel中间件的相关内容,可以阅读本专题下面的文章。

276

2024.04.09

laravel使用的设计模式有哪些
laravel使用的设计模式有哪些

laravel使用的设计模式有:1、单例模式;2、工厂方法模式;3、建造者模式;4、适配器模式;5、装饰器模式;6、策略模式;7、观察者模式。想了解更多laravel的相关内容,可以阅读本专题下面的文章。

370

2024.04.09

thinkphp和laravel哪个简单
thinkphp和laravel哪个简单

对于初学者来说,laravel 的入门门槛较低,更易上手,原因包括:1. 更简单的安装和配置;2. 丰富的文档和社区支持;3. 简洁易懂的语法和 api;4. 平缓的学习曲线。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

371

2024.04.10

laravel入门教程
laravel入门教程

本专题整合了laravel入门教程,想了解更多详细内容,请阅读专题下面的文章。

81

2025.08.05

laravel实战教程
laravel实战教程

本专题整合了laravel实战教程,阅读专题下面的文章了解更多详细内容。

64

2025.08.05

laravel面试题
laravel面试题

本专题整合了laravel面试题相关内容,阅读专题下面的文章了解更多详细内容。

67

2025.08.05

数据分析工具有哪些
数据分析工具有哪些

数据分析工具有Excel、SQL、Python、R、Tableau、Power BI、SAS、SPSS和MATLAB等。详细介绍:1、Excel,具有强大的计算和数据处理功能;2、SQL,可以进行数据查询、过滤、排序、聚合等操作;3、Python,拥有丰富的数据分析库;4、R,拥有丰富的统计分析库和图形库;5、Tableau,提供了直观易用的用户界面等等。

683

2023.10.12

html编辑相关教程合集
html编辑相关教程合集

本专题整合了html编辑相关教程合集,阅读专题下面的文章了解更多详细内容。

37

2026.01.21

热门下载

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

精品课程

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

共137课时 | 9.1万人学习

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

共6课时 | 9.5万人学习

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

共13课时 | 0.9万人学习

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

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