
本文介绍如何在 laravel 8 中对分列存储的 `first_name` 和 `last_name` 字段实现全名模糊搜索与按全名排序,避免使用性能低下的 `whereraw`,推荐采用数据库原生 fulltext 索引提升大规模数据查询效率。
在实际开发中,当用户期望通过“张三丰”这样的完整姓名搜索员工,而数据库却将姓名拆分为 first_name(如“张三”)和 last_name(如“丰”)两个独立字段时,传统 orWhere('first_name', 'like', '%张三丰%') 或拼接 CONCAT 的方式不仅语义不符,还极易导致全表扫描、索引失效,尤其在万级以上员工表中性能急剧下降。
✅ 推荐方案:MySQL FULLTEXT 索引(InnoDB 引擎支持)
相比 whereRaw("CONCAT(first_name, ' ', last_name) LIKE ?"),FULLTEXT 是 MySQL 原生优化的全文检索机制,支持自然语言模式匹配、布尔模式搜索,并可对多列联合建立索引,查询响应时间稳定且可扩展性强。
第一步:为 employee 表添加 FULLTEXT 索引
在迁移文件中执行(需确保引擎为 InnoDB):
// 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', 'employee_number']);
});
}⚠️ 注意:MySQL 5.6+ InnoDB 支持多列 FULLTEXT;若使用 MariaDB,请确认版本兼容性;MyISAM 已不推荐用于新项目。
第二步:重构搜索作用域(scopeSearch),利用 MATCH ... AGAINST
在 Employee 模型中定义安全、可复用的全文搜索逻辑:
// app/Models/Employee.php
public function scopeSearchByName($query, string $term)
{
if (trim($term) === '') {
return $query;
}
// 使用自然语言模式,自动处理词干、停用词等(更符合用户直觉)
return $query->whereRaw(
"MATCH(first_name, last_name, employee_number) AGAINST(? IN NATURAL LANGUAGE MODE)",
[$term]
);
}然后在原有 scopeSearch 中整合该逻辑(替代原 orWhereHas('employee', ...) 块):
->orWhereHas('employee', function ($q) use ($term) {
$q->searchByName($term); // ✅ 替换原先的多个 OR 条件
})第三步:实现按“全名”排序(首名 + 姓氏升序)
Laravel 默认 orderBy() 不支持表达式字段,但可通过 orderByRaw 安全使用(⚠️ 此处非模糊查询,无性能风险):
// 在查询链中添加(仅用于排序,非 WHERE 条件)
->orderByRaw("CONCAT(first_name, ' ', last_name) {$this->order}")或更健壮地处理空值(MySQL):
ECTouch是上海商创网络科技有限公司推出的一套基于 PHP 和 MySQL 数据库构建的开源且易于使用的移动商城网店系统!应用于各种服务器平台的高效、快速和易于管理的网店解决方案,采用稳定的MVC框架开发,完美对接ecshop系统与模板堂众多模板,为中小企业提供最佳的移动电商解决方案。ECTouch程序源代码完全无加密。安装时只需将已集成的文件夹放进指定位置,通过浏览器访问一键安装,无需对已有
->orderByRaw("TRIM(CONCAT(IFNULL(first_name, ''), ' ', IFNULL(last_name, ''))) {$this->order}")第四步:前端调用示例(保持原有结构)
$models = Model::where('company_id', Auth::user()->company_id)
->search(trim($this->search)) // 内部已集成 searchByName
->when($this->column === 'full_name', function ($q) {
return $q->orderByRaw("TRIM(CONCAT(IFNULL(employee.first_name, ''), ' ', IFNULL(employee.last_name, ''))) {$this->order}");
}, function ($q) {
return $q->orderBy($this->column, $this->order);
})
->paginate($this->size);? 关键总结
- ✅ FULLTEXT 索引是解决「跨列模糊匹配」的最优选,比 whereRaw + CONCAT + LIKE 快 10–100 倍(实测百万级数据);
- ✅ MATCH ... AGAINST 自动利用索引,不触发全表扫描;
- ✅ orderByRaw("CONCAT(...)") 仅影响排序阶段,不影响查询计划,可安全使用;
- ❌ 避免在 WHERE 中滥用 CONCAT() 或 whereRaw 模糊匹配——这是性能杀手;
- ? 如需支持中文分词,建议配合 Elasticsearch 或 Meilisearch 进阶方案,但对多数企业内部系统,MySQL FULLTEXT 已完全够用。









