0

0

Eloquent中实现自定义条件列与多字段搜索策略

DDD

DDD

发布时间:2025-09-11 11:32:11

|

560人浏览过

|

来源于php中文网

原创

Eloquent中实现自定义条件列与多字段搜索策略

本教程深入探讨在Laravel Eloquent中如何基于多个字段创建自定义计算列,以及如何执行高效的多字段联合搜索。我们将详细介绍使用DB::raw在数据库层面构建条件列、利用Eloquent查询构建器实现复杂的搜索逻辑,并通过Accessors在应用层处理数据展示。文章旨在帮助开发者根据具体业务需求,灵活选择并应用最适合的解决方案。

laravel应用开发中,我们经常会遇到需要将模型中的多个字段合并为一个逻辑上的“自定义列”,或者根据特定条件从多个字段中选择一个值作为最终输出。例如,一个模型可能包含title和original_title两个字段,我们希望在搜索或展示时,优先使用title的值,如果title为空,则退而使用original_title。本文将详细介绍在eloquent中实现这一目标的几种策略。

1. 使用 DB::raw 构建数据库层面的自定义计算列

当需要一个在数据库查询结果中实际存在的、基于其他字段计算出的列时,DB::raw是最高效且功能最强大的方法。它允许你直接嵌入原生SQL语句到Eloquent查询中,从而利用数据库的强大功能进行复杂的数据处理,例如条件逻辑(CASE WHEN)或函数调用。

场景描述: 创建一个名为cool_title的自定义列,其值在title非空时取title,否则取original_title。

示例代码:

has('search_term')) {
            $searchTerm = '%' . $request->input('search_term') . '%';
            $activities->where(DB::raw('CASE 
                                            WHEN title IS NOT NULL AND title != \'\' THEN title 
                                            ELSE original_title 
                                        END'), 'LIKE', $searchTerm);
        }

        // 获取结果
        $result = $activities->get();

        // 遍历结果,每个Activity对象都会有一个 cool_title 属性
        foreach ($result as $activity) {
            echo "ID: {$activity->id}, Title: {$activity->cool_title}\n";
        }

        return $result;
    }
}

注意事项:

  • NULL与空字符串: 在SQL中,NULL和空字符串''是不同的。title = ''只检查空字符串,而title IS NOT NULL检查非NULL。为了更严谨,通常建议同时检查IS NOT NULL AND title != ''。
  • 性能: 数据库层面的计算通常效率较高,尤其是在处理大量数据时。
  • 可搜索/排序: 通过DB::raw创建的列可以在后续的where、orderBy等子句中直接使用,就像普通列一样。
  • 可读性: 相比纯Eloquent方法,原生SQL的可读性可能会略有下降,但对于复杂逻辑,其表达力更强。
  • addSelect vs select: 如果你只想在现有select的基础上添加一个自定义列,可以使用addSelect。如果需要完全控制select语句,则使用select并显式列出所有需要的字段(包括*如果需要所有原始字段)。

2. 使用 Eloquent 查询构建器实现多字段条件搜索

如果你的核心需求不是创建一个新的计算列,而仅仅是希望根据多个字段的值来筛选数据,Eloquent查询构建器提供了更“优雅”的方式来构建复杂的WHERE子句。

场景描述: 搜索一个关键词,该关键词可能存在于title或original_title中。

示例代码:

input('query'); // 假设搜索词从请求中获取

        if (empty($searchTerm)) {
            return Activity::all(); // 如果没有搜索词,返回所有
        }

        $activities = Activity::where(function ($query) use ($searchTerm) {
            // 搜索条件:title 包含搜索词 OR original_title 包含搜索词
            $query->where('title', 'LIKE', '%' . $searchTerm . '%')
                  ->orWhere('original_title', 'LIKE', '%' . $searchTerm . '%');
        })->get();

        // 另一种更精确的搜索逻辑,如果title为空,则只搜索original_title
        // 假设我们想要找到那些 'coolTitle' 匹配 searchTerm 的记录
        // 这种情况下,我们需要模拟 'CASE WHEN' 的逻辑
        $activitiesConditional = Activity::where(function ($query) use ($searchTerm) {
            // 情况1: title 非空且匹配搜索词
            $query->whereNotNull('title')
                  ->where('title', '!=', '')
                  ->where('title', 'LIKE', '%' . $searchTerm . '%');
        })->orWhere(function ($query) use ($searchTerm) {
            // 情况2: title 为空或NULL,且 original_title 匹配搜索词
            $query->where(function ($q) {
                      $q->whereNull('title')->orWhere('title', '');
                  })
                  ->where('original_title', 'LIKE', '%' . $searchTerm . '%');
        })->get();

        return [
            'simple_search_results' => $activities,
            'conditional_search_results' => $activitiesConditional
        ];
    }
}

注意事项:

  • 逻辑分组: 使用闭包 (function ($query) { ... }) 可以有效地对WHERE子句进行逻辑分组,生成如 (condition1 OR condition2) AND condition3 这样的复杂查询。
  • 不创建新列: 这种方法不会在结果集中添加一个名为cool_title的新列,它仅仅是影响了哪些记录会被检索出来。
  • 可读性: 纯Eloquent查询通常比DB::raw更具可读性和维护性。

3. 使用 Eloquent Accessors 处理模型层面的展示逻辑

如果自定义列的需求仅仅是为了在应用层(例如视图或API响应)展示数据,而不需要在数据库层面进行搜索、排序或聚合,那么Eloquent Accessors(访问器)是一个非常简洁优雅的解决方案。

场景描述: 在获取Activity模型实例后,为其添加一个cool_title属性用于显示。

微信小程序-康爱多商城
微信小程序-康爱多商城

微信小程序-康爱多商城是一个微信小程序的商城demo源码。它实现了商城的首页功能。软件特色:自定义搜索框图标grid水平拖动列表线条中间显示文字

下载

示例代码:

app/Models/Activity.php模型中定义访问器:

attributes['title'])) {
            return $this->attributes['title'];
        }

        // 如果 title 为空,则返回 original_title
        return $this->attributes['original_title'] ?? ''; // 使用 ?? 确保返回字符串
    }

    // 如果希望这个属性在模型被转换为数组或JSON时自动包含
    // protected $appends = ['cool_title'];
}

使用示例:

id}\n";
        echo "Display Title: {$activity->cool_title}\n"; // 会自动调用 getCoolTitleAttribute()

        // 如果在模型中设置了 $appends = ['cool_title'];
        // 那么 $activity->toArray() 或 json_encode($activity) 会包含 cool_title
        return $activity;
    }
}

注意事项:

  • 仅限应用层: Accessors在数据从数据库检索到PHP应用内存后才执行,因此无法用于数据库层面的WHERE、ORDER BY或GROUP BY子句。
  • 可读性与封装: 将显示逻辑封装在模型内部,使代码更清晰、更易于维护。
  • $appends属性: 如果希望访问器属性在模型被序列化为数组或JSON时自动包含,需要在模型中添加protected $appends = ['cool_title'];。

总结与选择建议

选择哪种方法取决于你的具体需求:

  1. 需要一个在数据库层面可搜索、可排序、可过滤的计算列?

    • 选择 DB::raw。 这是唯一能在数据库查询结果中创建新计算列的方法,适用于需要利用数据库索引或进行复杂聚合的场景。
  2. 只需要根据多个字段的值来筛选记录,不关心结果集中是否有新的计算列?

    • 选择 Eloquent 查询构建器的 where 和 orWhere 组合。 这种方法更“Eloquent化”,代码更简洁,易于维护。
  3. 自定义列仅用于前端展示,不需要在数据库层面进行任何操作?

    • 选择 Eloquent Accessors。 它将显示逻辑封装在模型中,保持了模型的清洁和可读性,但牺牲了数据库层面的功能性。

在实际开发中,开发者应根据业务场景和性能要求,灵活运用上述策略,以实现代码的优雅与功能的强大。通常,如果数据库能够完成的工作,尽量让数据库去完成,因为它在处理大量数据时通常更高效。但如果逻辑过于复杂,或者仅影响显示,那么在应用层处理也是一个不错的选择。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

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

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

2911

2023.09.01

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

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

1736

2023.10.11

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

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

1567

2023.10.11

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

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

1120

2023.10.23

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

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

1566

2023.10.23

html怎么上传
html怎么上传

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

1297

2023.11.03

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

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

1669

2023.11.09

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

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

1310

2023.11.13

拼多多赚钱的5种方法 拼多多赚钱的5种方法
拼多多赚钱的5种方法 拼多多赚钱的5种方法

在拼多多上赚钱主要可以通过无货源模式一件代发、精细化运营特色店铺、参与官方高流量活动、利用拼团机制社交裂变,以及成为多多进宝推广员这5种方法实现。核心策略在于通过低成本、高效率的供应链管理与营销,利用平台社交电商红利实现盈利。

31

2026.01.26

热门下载

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

精品课程

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

共137课时 | 9.5万人学习

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

共6课时 | 11.2万人学习

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

共13课时 | 0.9万人学习

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

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