【thinkphp知识】thinkphp 使用union链式操作合并两个或多个 SELECT 语句的结果集,新增不存在的字段,并赋予默认值

简介: 【thinkphp知识】thinkphp 使用union链式操作合并两个或多个 SELECT 语句的结果集,新增不存在的字段,并赋予默认值

前言

之前有做过laravel使用union的:laravelDb查询数据库使用union合并两个不相关的表,新增不存在的字段,并赋予默认值

那么thinkphp又要如何做呢?

我们可以先查看官方文档:https://static.kancloud.cn/manual/thinkphp5/118084

使用案例

Db::field('name')
   ->table('think_user_0')
   ->union(function($query){
    $query->field('name')->table('think_user_1');
     })
   ->union(function($query){
    $query->field('name')->table('think_user_2');
     })
   ->select();

当然,官方的这种办法是可以实现的,但是明显还不够优雅,我们可以结合buildSql构造子查询一起来使用:https://static.kancloud.cn/manual/thinkphp5/135185

使用例子

$subQuery = Db::table('think_user')
    ->field('id,name')
    ->where('id','>',10)
    ->buildSql();

实际案例

需求分析

今天有一个需求,需要写一个全局搜索接口,系统的数据很多,包括新闻,文章,协会专家,供应采购等等数据,覆盖多个表的数据,每部分的内容都太一样,当然你可以直接简单粗暴的使用遍历查询每个表的数据,再进行合并,但是这样明显不够方便,开销也大,而且肯定还会涉及数据分页,手写分页也是挺麻烦的,所以我们可以选择使用union直接一步到位,UNION操作就是用于合并两个或多个 SELECT 语句的结果集。

几个数据库表结构

代码实现

'' AS title表示新增不存在的字段,并赋予默认值为空字符

//全局搜索接口
public function search()
{
    $request = request()->post();
    $limit = $request['limit'] ?? 10;
    $title = $request['title'] ?? '';

    //菜单导航:1=新闻中心,2=网上办事,3=法规标准,4=协会专家,5=供应,6=采购
    $where = [];
    if ($title) $where['name'] = ['like', "%" . $title . "%"];
    $sql4 = DB::name("expert")->field("id, name, '' AS title, tags, '' AS url , FROM_UNIXTIME(createtime) as createtime, 4 AS menu")
        ->where($where)
        ->buildSql();
    $sql5 =DB::name("supply")->field("id, name, '' AS title, tags, '' AS url , FROM_UNIXTIME(createtime) as createtime, 5 AS menu")
        ->where($where)
        ->buildSql();

    $where = [];
    if ($title) $where['title'] = ['like', "%" . $title . "%"];
    $sql1 = DB::name("news")->field("id, '' AS name, title, tags, '' AS url , FROM_UNIXTIME(createtime) as createtime, 1 AS menu")
        ->where($where)
        ->buildSql();
    $sql2 = DB::name("work")->field("id, '' AS name, title, tags, url, FROM_UNIXTIME(createtime) as createtime, 2 AS menu")
        ->where($where)
        ->buildSql();
    $sql3 = DB::name("laws")->field("id, '' AS name, title, tags, '' AS url , FROM_UNIXTIME(createtime) as createtime, 3 AS menu")
        ->where($where)
        ->buildSql();
    $sql6 = DB::name("purchase")->field("id, '' AS name, title, tags, '' AS url, FROM_UNIXTIME(createtime) as createtime, 6 AS menu") 
        ->where($where);
    //合并
    $sql = $sql6->union([$sql1, $sql2, $sql3, $sql4, $sql5])->buildSql();
    $data = Db::table($sql.' a')->order("a.createtime desc")->paginate($limit);
        $this->success('获取成功', $data);
    }

结果

注意

UNION 内部的 SELECT 语句必须拥有相同数量的列。列也必须拥有相似的数据类型。同时,每条 SELECT 语句中的列的顺序必须相同

UNION与UNION ALL的区别

当使用 UNION 时,MySQL 会把结果集中重复的记录删掉,而使用 UNION ALL ,MySQL 会把所有的记录返回,且效率高于 UNION。

想使用UNION ALL也非常简单

Db::field('name')
    ->table('think_user_0')
    ->unionAll('SELECT name FROM think_user_1')
    ->unionAll('SELECT name FROM think_user_2')
    ->select();

或者

Db::field('name')
    ->table('think_user_0')
    ->union(['SELECT name FROM think_user_1', 'SELECT name FROM think_user_2'], true)
    ->select();
目录
相关文章
|
7月前
|
SQL 关系型数据库 MySQL
MYSQL根据查询结果删除sql 去除重复id 新增对比前一条与后一条数据 去重3种方法​ 窗口函数
MYSQL根据查询结果删除sql 去除重复id 新增对比前一条与后一条数据 去重3种方法​ 窗口函数
145 0
|
4月前
|
SQL 数据挖掘 数据库
|
7月前
|
SQL 关系型数据库 MySQL
【MySQL】DQL-基础查询-语句&演示(查询多个字段 / 所有字段/并设置别名/去重)
【MySQL】DQL-基础查询-语句&演示(查询多个字段 / 所有字段/并设置别名/去重)
|
7月前
|
SQL 关系型数据库 MySQL
【MySQL】DQL-排序查询-语法&排序方式&注意事项&可cv例题语句
【MySQL】DQL-排序查询-语法&排序方式&注意事项&可cv例题语句
|
存储
PG11新特性解读:新增非空默认值字段不需要重写表
PG11新特性解读:新增非空默认值字段不需要重写表
192 1
|
7月前
|
SQL Oracle 关系型数据库
sql语句两个字段合并或者两个字段拼接显示
sql语句两个字段合并或者两个字段拼接显示
|
7月前
|
SQL 存储 Java
MyBatis【付诸实践 02】 mapper文件未编译+statementType使用+返回结果字段顺序不一致+获取自增ID+一个update标签批量更新记录
MyBatis【付诸实践 02】 mapper文件未编译+statementType使用+返回结果字段顺序不一致+获取自增ID+一个update标签批量更新记录
88 0
|
7月前
|
SQL 数据库
SQL标识列实现自动编号的步骤和技巧以及优势
SQL标识列实现自动编号的步骤和技巧以及优势
152 0
|
SQL 数据库管理
【SQL开发实战技巧】系列(九):一个update误把其他列数据更新成空了?Merge改写update!给你五种删除重复数据的写法!
本篇文章讲解的主要内容是:***你有没有经历过一个update把其他列数据清空了、使用merge更新合并记录、删除违反参照完整性的记录、给你五种删除重复数据的写法*** 【SQL开发实战技巧】这一系列博主当作复习旧知识来进行写作,毕竟SQL开发在数据分析场景非常重要且基础,面试也会经常问SQL开发和调优经验,相信当我写完这一系列文章,也能再有所收获,未来面对SQL面试也能游刃有余~。
【SQL开发实战技巧】系列(九):一个update误把其他列数据更新成空了?Merge改写update!给你五种删除重复数据的写法!
|
SQL 大数据 开发者
SQL 语法--表特定语句--分组、排序、过滤 | 学习笔记
快速学习 SQL 语法--表特定语句--分组、排序、过滤
254 0
SQL 语法--表特定语句--分组、排序、过滤 | 学习笔记

相关课程

更多