0

0

如何在 Laravel 中高效实现全名搜索与排序(无需 whereRaw)

心靈之曲

心靈之曲

发布时间:2026-02-03 20:55:01

|

656人浏览过

|

来源于php中文网

原创

如何在 Laravel 中高效实现全名搜索与排序(无需 whereRaw)

本文介绍在 laravel 8 中不使用 `whereraw` 的前提下,通过数据库原生全文索引(fulltext)对分散存储的 `first_name` 和 `last_name` 字段进行联合模糊搜索与按全名排序的高性能方案。

在实际开发中,当用户姓名被拆分为 first_name 和 last_name 两个独立字段时,常见的搜索需求(如输入“John Doe”匹配完整姓名)和排序需求(如 orderBy('full_name'))往往难以优雅实现——若强行拼接字符串或使用 whereRaw(CONCAT(...)),不仅破坏 Eloquent 的可移植性,更会在大数据量场景下引发严重性能瓶颈(如全表扫描、无法利用索引)。

推荐方案:MySQL FULLTEXT 索引 + Laravel 原生查询构造器

MySQL 的 FULLTEXT 索引专为文本搜索优化,支持自然语言模式(MATCH ... AGAINST),可跨多列建立联合索引,并自动处理词序、停用词与相关性排序。相比 LIKE '%term%' 或 CONCAT() 拼接,其查询速度提升可达数量级,且完全避免 whereRaw 的安全与维护风险。

第一步:为 first_name 和 last_name 添加联合 FULLTEXT 索引
在迁移文件中添加:

// php artisan make:migration add_fulltext_index_to_employees
public function up(MigrationBuilder $migration)
{
    $migration->table('employees', function (Blueprint $table) {
        $table->fullText(['first_name', 'last_name']);
    });
}

执行迁移后,MySQL 将创建一个覆盖两列的全文索引(注意:仅 InnoDB 表支持多列 FULLTEXT;MyISAM 已弃用)。

第二步:重构 scopeSearch,使用 MATCH ... AGAINST 替代 orWhere 链

public function scopeSearch($query, $term)
{
    $term = trim($term);
    if (empty($term)) {
        return $query;
    }

    // 使用自然语言模式(默认),自动按相关性排序
    $query->where(function ($q) use ($term) {
        $q->where('type', 'LIKE', "%{$term}%")
          ->orWhere('name', 'LIKE', "%{$term}%")
          ->orWhereHas('level', fn($sub) => $sub->where('name', 'LIKE', "%{$term}%"))
          ->orWhereHas('employee', function ($sub) use ($term) {
              // ✅ 关键替换:用 MATCH 对 first_name + last_name 联合搜索
              $sub->whereRaw("MATCH(first_name, last_name) AGAINST(? IN NATURAL LANGUAGE MODE)", [$term]);
              // 同时保留 employee_number 等其他字段的 LIKE 搜索
              $sub->orWhere('employee_number', 'LIKE', "%{$term}%");
          })
          ->orWhereHas('status', fn($sub) => $sub->where('name', 'LIKE', "%{$term}%"));
    });

    return $query;
}

⚠️ 注意:MATCH ... AGAINST 在 NATURAL LANGUAGE MODE 下不支持通配符(如 *),但会智能匹配词干与近义词;若需精确前缀匹配(如“Joh”),可改用 BOOLEAN MODE 并配合 `+John +Doe*` 语法(需确保最小词长配置合理)。

第三步:实现按全名排序(无需虚拟列或 whereRaw)

AlibabaWOOD
AlibabaWOOD

阿里巴巴打造的多元电商视频智能创作平台

下载

由于 MySQL 不支持直接 ORDER BY CONCAT(first_name, ' ', last_name) 利用索引,但可通过 MATCH 的隐式相关性排序获得“语义上更匹配”的结果。若需严格字典序排序(如 A-Z 全名列表),建议:

  • 方案 A(推荐):添加生成列(Generated Column)并建索引(MySQL 5.7+)

    ALTER TABLE employees 
      ADD COLUMN full_name VARCHAR(255) 
        GENERATED ALWAYS AS (CONCAT(TRIM(first_name), ' ', TRIM(last_name))) STORED,
      ADD INDEX idx_full_name (full_name);

    Laravel 中即可安全使用:

    ->orderBy('full_name', $this->order)
  • 方案 B(兼容性更强):应用层排序(适用于分页数据量可控场景)

    $models = Model::where('company_id', Auth::user()->company_id)
        ->search(trim($this->search))
        ->paginate($this->size);
    
    // 对当前页结果按全名排序(注意:仅限小页码,避免内存溢出)
    $models->items()->sort(function ($a, $b) {
        $fullNameA = trim("{$a->employee->first_name} {$a->employee->last_name}");
        $fullNameB = trim("{$b->employee->first_name} {$b->employee->last_name}");
        return $this->order === 'desc' 
            ? strcmp($fullNameB, $fullNameA) 
            : strcmp($fullNameA, $fullNameB);
    });

? 关键总结

  • ✅ FULLTEXT 是解决多字段联合模糊搜索的标准、高效、安全方案,远优于 whereRaw 或 LIKE;
  • ✅ 生成列(STORED)+ 索引是实现 ORDER BY full_name 的最佳实践,兼顾性能与可维护性;
  • ❌ 避免在 WHERE 或 ORDER BY 中使用 CONCAT()、TRIM() 等函数,会导致索引失效;
  • ? 测试时务必使用 EXPLAIN 验证查询是否命中 FULLTEXT 或 idx_full_name 索引。

通过以上改造,你将获得毫秒级全名搜索响应与稳定的字典序排序能力,同时保持 Laravel 代码的清晰性与数据库层的高性能。

热门AI工具

更多
DeepSeek
DeepSeek

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

豆包大模型
豆包大模型

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

通义千问
通义千问

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

腾讯元宝
腾讯元宝

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

文心一言
文心一言

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

讯飞写作
讯飞写作

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

即梦AI
即梦AI

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

ChatGPT
ChatGPT

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

相关专题

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

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

321

2024.04.09

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

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

281

2024.04.09

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

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

435

2024.04.09

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

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

374

2024.04.10

laravel入门教程
laravel入门教程

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

86

2025.08.05

laravel实战教程
laravel实战教程

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

72

2025.08.05

laravel面试题
laravel面试题

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

68

2025.08.05

mysql修改数据表名
mysql修改数据表名

MySQL修改数据表:1、首先查看数据库中所有的表,代码为:‘SHOW TABLES;’;2、修改表名,代码为:‘ALTER TABLE 旧表名 RENAME [TO] 新表名;’。php中文网还提供MySQL的相关下载、相关课程等内容,供大家免费下载使用。

669

2023.06.20

全国统一发票查询平台入口合集
全国统一发票查询平台入口合集

本专题整合了全国统一发票查询入口地址合集,阅读专题下面的文章了解更多详细入口。

19

2026.02.03

热门下载

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

精品课程

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

共48课时 | 2.1万人学习

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

共3课时 | 0.3万人学习

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

共1课时 | 821人学习

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

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