0

0

Eloquent 中实现条件字段选择与搜索:告别原始 SQL 的优雅之道

霞舞

霞舞

发布时间:2025-09-11 12:03:01

|

462人浏览过

|

来源于php中文网

原创

Eloquent 中实现条件字段选择与搜索:告别原始 SQL 的优雅之道

本文探讨在 Laravel Eloquent 中如何优雅地实现条件字段选择,即根据 title 和 original_title 字段的值动态生成一个新字段。我们将介绍两种主要方法:利用 Eloquent Accessor 进行模型层面的处理,以及在必要时使用 DB::raw 进行数据库层面的计算。此外,还将涵盖如何在这些字段上执行条件搜索。

在开发过程中,我们经常会遇到需要根据现有字段的条件逻辑来生成一个新字段的需求。例如,在一个模型中存在 title 和 original_title 两个字段,我们希望在查询结果中获得一个统一的 cooltitle 字段,其值优先取 title,如果 title 为空(或 null),则取 original_title。尽管使用 db::raw 可以直接在数据库层面实现这一逻辑,但有时我们希望有更“eloquent 风格”的解决方案。

一、使用 Eloquent Accessor 实现模型层面的条件字段选择

Eloquent Accessor(访问器)是 Laravel 提供的一种优雅方式,用于在模型实例被访问时自动修改或计算属性。它非常适合在数据从数据库取出后进行二次处理,以生成新的派生属性,而无需修改原始数据库查询。

1. 定义 Accessor

在您的 Activity 模型中,定义一个名为 getCoolTitleAttribute 的方法。Eloquent 会自动将 cool_title 属性的访问映射到这个方法。

// app/Models/Activity.php

namespace App\Models;

use Illuminate\Database\Eloquent\Factories\HasFactory;
use Illuminate\Database\Eloquent\Model;

class Activity extends Model
{
    use HasFactory;

    /**
     * 获取活动的统一标题 (coolTitle)。
     * 如果 title 存在且非空,则使用 title;否则使用 original_title。
     *
     * @return string|null
     */
    public function getCoolTitleAttribute(): ?string
    {
        // 这里的 'title' 和 'original_title' 是模型实例的属性
        // 可以根据实际情况判断空字符串或 null
        return $this->attributes['title'] ?? $this->attributes['original_title'];
        // 或者更严格的空字符串检查:
        // return !empty($this->attributes['title']) ? $this->attributes['title'] : $this->attributes['original_title'];
    }

    /**
     * 如果希望在模型序列化为 JSON 时自动包含此属性,
     * 可以将其添加到 $appends 数组中。
     *
     * @var array
     */
    protected $appends = ['cool_title'];
}

2. 使用 Accessor

一旦定义了 Accessor,您就可以像访问模型其他属性一样访问 cool_title。

$activity = Activity::find(1);

// 访问 coolTitle 属性
echo $activity->cool_title; // 会自动调用 getCoolTitleAttribute 方法

// 如果查询结果是集合
$activities = Activity::all();
foreach ($activities as $activity) {
    echo $activity->cool_title . "\n";
}

// 当模型被序列化为 JSON 时(例如在 API 响应中),
// 如果 $appends 数组中包含了 'cool_title',它也会被自动包含。
return response()->json($activity);

优点:

  • 代码整洁: 将业务逻辑封装在模型内部,符合面向对象原则。
  • 可读性强: 避免了复杂的 SQL 片段,提高了代码可读性
  • 易于维护: 逻辑集中管理,方便修改和测试。

缺点:

  • 性能开销: 字段计算发生在数据从数据库取出后,对于大量数据或需要高性能的场景,可能不如数据库层面计算高效。
  • 无法直接用于数据库查询/排序: 您不能直接在 where('cool_title', '...') 或 orderBy('cool_title') 中使用此属性,因为 cool_title 不存在于数据库表中。

二、使用 DB::raw 实现数据库层面的条件字段选择

尽管用户可能倾向于避免原始 SQL,但在某些场景下,例如需要在数据库层面进行高效过滤、排序,或者处理大量数据时,DB::raw 仍然是不可替代的强大工具。它允许您直接将 SQL 片段注入到 Eloquent 查询中。

1. 构建查询

我们可以使用 addSelect 方法结合 DB::raw 来添加一个计算字段。

use Illuminate\Support\Facades\DB;
use App\Models\Activity;

// 假设我们正在查询 Activity 模型
$activities = Activity::addSelect([
    'id', // 选择其他需要的字段
    'title',
    'original_title',
    DB::raw('(CASE WHEN title IS NULL OR title = "" THEN original_title ELSE title END) as coolTitle')
])->get();

foreach ($activities as $activity) {
    echo "ID: {$activity->id}, Title: {$activity->title}, Original Title: {$activity->original_title}, Cool Title: {$activity->coolTitle}\n";
}

注意事项:

  • CASE WHEN 语句中的条件需要根据实际情况调整。IS NULL 用于检查 NULL 值,title = "" 用于检查空字符串。通常建议同时检查。
  • addSelect 方法会保留您模型中 $fillable 或 $guarded 之外的其他字段,但为了清晰起见,最好明确列出所有需要的字段。

2. 使用数据库函数简化 (例如 MySQL 的 IFNULL/COALESCE)

微信 WeLM
微信 WeLM

WeLM不是一个直接的对话机器人,而是一个补全用户输入信息的生成模型。

下载

如果您的数据库支持,可以使用更简洁的数据库函数,例如 MySQL 的 IFNULL 或 COALESCE。COALESCE 函数返回其参数中第一个非 NULL 的表达式。

// 使用 COALESCE,它会返回第一个非 NULL 的表达式
// 如果 title 可能为空字符串但不是 NULL,则需要更复杂的逻辑
$activities = Activity::addSelect([
    'id',
    'title',
    'original_title',
    DB::raw('COALESCE(NULLIF(title, ""), original_title) as coolTitle')
])->get();
// NULLIF(title, "") 会将空字符串的 title 转换为 NULL,
// 然后 COALESCE 就可以处理 NULL 和 original_title。

// 如果只需要处理 NULL 值,COALESCE 足够
$activities = Activity::addSelect([
    'id',
    'title',
    'original_title',
    DB::raw('COALESCE(title, original_title) as coolTitle')
])->get();

优点:

  • 性能高效: 计算在数据库层面完成,减少了应用层面的处理负担。
  • 可用于查询/排序: 可以直接在 where('coolTitle', '...') 或 orderBy('coolTitle') 中使用这个计算字段。

缺点:

  • 可读性略低: 相比 Accessor,SQL 片段可能略显复杂。
  • 数据库依赖: DB::raw 中的 SQL 语法可能依赖于特定的数据库系统(例如 MySQL 的 IFNULL,PostgreSQL 的 COALESCE)。

三、处理基于条件字段的搜索

原始问题中也提到了“make a search”的需求。如果您需要根据 title 或 original_title 的值进行搜索,而不一定需要创建一个新的 coolTitle 字段,那么可以使用 where 和 orWhere 组合。

1. 简单的条件搜索

假设您要搜索 $search 关键字,如果 title 不为空且匹配,或者 original_title 匹配,则返回结果。

use App\Models\Activity;

$search = 'some keyword';

$activities = Activity::where(function ($query) use ($search) {
    // 检查 title 是否不为空且匹配搜索词
    $query->whereNotNull('title')
          ->where('title', 'like', '%' . $search . '%');
})->orWhere(function ($query) use ($search) {
    // 或者 original_title 匹配搜索词
    $query->where('original_title', 'like', '%' . $search . '%');
})->get();

2. 考虑空值/空字符串的搜索逻辑

如果您的逻辑是“如果 title 为空,则搜索 original_title;否则搜索 title”,这在单个数据库查询中实现会稍微复杂,通常需要 DB::raw 或更复杂的 where 嵌套。

例如,如果您想搜索 coolTitle 的逻辑(title 优先,否则 original_title):

use Illuminate\Support\Facades\DB;
use App\Models\Activity;

$search = 'some keyword';

$activities = Activity::where(function ($query) use ($search) {
    // 如果 title 不为空,则搜索 title
    $query->whereNotNull('title')
          ->where('title', 'like', '%' . $search . '%');
})->orWhere(function ($query) use ($search) {
    // 如果 title 为空(或 null),则搜索 original_title
    $query->where(function ($subQuery) {
        $subQuery->whereNull('title')->orWhere('title', ''); // 检查 title 为 null 或空字符串
    })->where('original_title', 'like', '%' . $search . '%');
})->get();

这种方法虽然没有直接创建 coolTitle 字段,但实现了基于相同逻辑的搜索功能。

四、选择最佳方案

  • 如果主要目的是在模型实例被访问时显示或处理派生属性,并且不需要在数据库层面进行过滤或排序,推荐使用 Eloquent Accessor。 它提供了最“Eloquent 风格”的解决方案,代码整洁且易于维护。
  • 如果需要根据派生字段进行高效的数据库过滤、排序,或者处理大量数据时性能至关重要,那么使用 DB::raw 是更合适的选择。 尽管涉及原始 SQL,但它直接利用了数据库的强大功能。
  • 如果仅仅是需要在两个字段中进行条件搜索,而不需要生成一个新字段,则可以使用 where 和 orWhere 的组合来构建复杂的搜索逻辑。

最终的选择取决于您的具体需求、对性能的考量以及对代码风格的偏好。通常,从 Accessor 开始,如果遇到性能瓶颈或需要数据库层面的复杂操作,再考虑转向 DB::raw 是一个明智的策略。

热门AI工具

更多
DeepSeek
DeepSeek

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

豆包大模型
豆包大模型

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

通义千问
通义千问

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

腾讯元宝
腾讯元宝

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

文心一言
文心一言

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

讯飞写作
讯飞写作

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

即梦AI
即梦AI

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

ChatGPT
ChatGPT

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

相关专题

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

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

319

2024.04.09

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

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

277

2024.04.09

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

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

371

2024.04.09

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

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

373

2024.04.10

laravel入门教程
laravel入门教程

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

81

2025.08.05

laravel实战教程
laravel实战教程

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

65

2025.08.05

laravel面试题
laravel面试题

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

68

2025.08.05

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

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

706

2023.10.12

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

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

31

2026.01.26

热门下载

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

精品课程

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

共48课时 | 1.9万人学习

MySQL 初学入门(mosh老师)
MySQL 初学入门(mosh老师)

共3课时 | 0.3万人学习

简单聊聊mysql8与网络通信
简单聊聊mysql8与网络通信

共1课时 | 811人学习

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

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