【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();
目录
相关文章
|
8月前
|
SQL 关系型数据库 MySQL
MYSQL根据查询结果删除sql 去除重复id 新增对比前一条与后一条数据 去重3种方法​ 窗口函数
MYSQL根据查询结果删除sql 去除重复id 新增对比前一条与后一条数据 去重3种方法​ 窗口函数
149 0
|
5月前
|
SQL 数据挖掘 数据库
|
8月前
|
SQL 关系型数据库 MySQL
【MySQL】DQL-基础查询-语句&演示(查询多个字段 / 所有字段/并设置别名/去重)
【MySQL】DQL-基础查询-语句&演示(查询多个字段 / 所有字段/并设置别名/去重)
|
8月前
|
SQL Oracle 关系型数据库
sql语句两个字段合并或者两个字段拼接显示
sql语句两个字段合并或者两个字段拼接显示
|
8月前
|
SQL 存储 Java
MyBatis【付诸实践 02】 mapper文件未编译+statementType使用+返回结果字段顺序不一致+获取自增ID+一个update标签批量更新记录
MyBatis【付诸实践 02】 mapper文件未编译+statementType使用+返回结果字段顺序不一致+获取自增ID+一个update标签批量更新记录
90 0
|
8月前
|
SQL 数据库
SQL标识列实现自动编号的步骤和技巧以及优势
SQL标识列实现自动编号的步骤和技巧以及优势
153 0
|
SQL 大数据 开发者
SQL 语法--表特定语句--分组、排序、过滤 | 学习笔记
快速学习 SQL 语法--表特定语句--分组、排序、过滤
264 0
SQL 语法--表特定语句--分组、排序、过滤 | 学习笔记
|
索引 数据库
SqlServer中的UNION操作符在合并数据时去重的原理以及UNION运算符查询结果默认排序的问题
原文:SqlServer中的UNION操作符在合并数据时去重的原理以及UNION运算符查询结果默认排序的问题   本文出处:http://www.cnblogs.com/wy123/p/7884986.
2292 0
|
SQL 存储 Java
mybatis如何直接 执行传入的任意sql语句 并按照顺序取出查询的结果集
mybatis如何直接 执行传入的任意sql语句 并按照顺序取出查询的结果集 需求: 1.直接执行前端传来的任何sql语句,parameterType="String", 2.对于任何sql语句,其返回值类型无法用resultMap在xml文件里配置或者返回具体的bean类型,因此设置resultType="java.util.Map",但是Map并不保证存入取出顺序一致, 因此设置resultType="java.util.LinkedHashMap",为保证查询的字段值有序(存入与取出顺序一致)所以采用LinkedHashMap。
2483 0
|
关系型数据库 MySQL Java
mysql字段的细节(查询自定义的字段[意义:行列转置];UNION ALL;case-when)
mysql字段的细节(查询自定义的字段[意义:行列转置];UNION ALL;case-when)
212 0
mysql字段的细节(查询自定义的字段[意义:行列转置];UNION ALL;case-when)