thinkphp6入门(25)-- 分组查询 GROUP_CONCAT

news/2024/10/3 11:33:49 标签: mysql, sql, 数据库, thinkphp

假设表名为 user_courses,字段为 user_id 和 course_name,存储每个用户选修的课程,想查询每个学生选修的所有课程

图片

SQL 原生查询

sql">SELECT user_id, GROUP_CONCAT(course_name) as courses FROM user_courses GROUP BY user_id;
ThinkPHP 代码实现

在 ThinkPHP 中,你可以使用查询构造器来实现类似的功能:


use think\facade\Db;

// 查询每个用户选修的所有课程
$data = Db::name('user_courses')
    ->field('user_id, GROUP_CONCAT(course_name) as courses') // 使用 GROUP_CONCAT 函数将课程合并
    ->group('user_id') // 按 user_id 分组
    ->select();

// 输出结果
print_r($data);

解释:

  1. field('user_id, GROUP_CONCAT(course_name) as courses'):选择 user_id 和使用 GROUP_CONCAT 函数将 course_name 字段中该用户的所有课程组合成一个字符串,并命名为 courses

  2. group('user_id'):按 user_id 进行分组,确保每个用户只会有一行结果,并且课程会聚合到这一行中。

也可以使用模型查询

UserCourse.php

<?php
namespace app\model;

use think\Model;

class UserCourse extends Model
{
    // 设置当前模型对应的完整数据表名称
    protected $name = 'user_courses'; // 表名
}

use app\model\UserCourse;

$data = UserCourse::field('user_id, GROUP_CONCAT(course_name) as courses')  // 聚合查询
    ->group('user_id')  // 按用户分组
    ->select();

// 输出查询结果
print_r($data);

查询结果

返回的数据格式类似如下:

每个用户的 courses 字段是该用户选修的所有课程的组合,多个课程以逗号分隔。


[
    ['user_id' => 1, 'courses' => 'Math,English,History'],
    ['user_id' => 2, 'courses' => 'Physics,Chemistry'],
    ['user_id' => 3, 'courses' => 'Biology,Geography,Math']
]

扩展:

如果你想按字母顺序排列每个用户的课程,可以在 GROUP_CONCAT 中使用 ORDER BY 进行排序:

use think\facade\Db;

$data = Db::name('user_courses')
    ->field('user_id, GROUP_CONCAT(course_name ORDER BY course_name ASC) as courses')
    ->group('user_id')
    ->select();

print_r($data);

如果结果想以user_id为key

<?php
namespace app\controller;

use app\BaseController;
use app\model\UserCourse;

class Course extends BaseController
{
    public function index()
{
        // 查询每个用户选修的所有课程
        $data = UserCourse::field('user_id, GROUP_CONCAT(course_name) as courses')
            ->group('user_id')
            ->select()
            ->toArray();

        // 使用 array_column 将 user_id 作为键
        $data = array_column($data, null, 'user_id');

        // 返回 JSON 数据
        return json($data);
    }
}

结果


[
    1 => ['user_id' => 1, 'courses' => 'Math,English,History'],
    2 => ['user_id' => 2, 'courses' => 'Physics,Chemistry'],
    3 => ['user_id' => 3, 'courses' => 'Biology,Geography,Math'],
]

如果想将字符串转为数组

// 将 courses 字段的字符串转换为数组
foreach ($data as &$item) {
     $item['courses'] = explode(',', $item['courses']);
}

by 软件工程小施同学


http://www.niftyadmin.cn/n/5688272.html

相关文章

Bilibili视频如何保存到本地

Bilibili(哔哩哔哩)作为中国领先的视频分享平台之一&#xff0c;汇聚了大量的优质内容&#xff0c;从搞笑动画、综艺节目到专业教程&#xff0c;应有尽有。许多用户时常会遇到这样的需求&#xff1a;希望将视频保存到本地&#xff0c;方便离线观看或者保存珍藏。由于版权保护等…

滚雪球学Oracle[3.4讲]:事务控制与锁管理

全文目录&#xff1a; 前言一、事务隔离级别的深入探讨1.1 事务的定义与基本特性1.2 事务隔离级别的概念1.3 各隔离级别中的问题案例演示&#xff1a;不同隔离级别的行为 1.4 隔离级别与性能的权衡 二、锁的种类与死锁问题解决2.1 锁的种类2.2 锁的粒度2.3 死锁与解决策略死锁的…

三、数据链路层(上)

目录 3.1数据链路层概述 3.1.1术语 3.1.2功能 3.2封装成帧和透明传输 3.2.1封装成帧 ①字符计数法 ②字符&#xff08;节&#xff09;填充法 ③零比特填充法 ④违规编码法 3.2.2透明传输 3.2.3差错控制 差错原因 检错编码 奇偶校验 ☆循环冗余码CRC 例题 纠错…

`git fetch` 检查更新

git fetch 是 Git 中的一个命令&#xff0c;主要用于从远程仓库获取最新的更新&#xff0c;但不会自动将这些更新合并到你的本地分支。它的主要作用是让你可以查看远程仓库的最新变化&#xff0c;而不改变你当前正在工作的代码。 1. 获取远程更新 git fetch 会从远程仓库下载…

python之with

with上下文管理是什么呢&#xff1f; 一般都是使用系统提供的一些with语句&#xff0c;列如我要去读取一些数据进行分析&#xff0c;就可以使用with open去读取某些数据&#xff0c;或者我要把一些图片给他保存到某些地方&#xff0c;可以用with给他写入。 上下午管理器with是…

DAY84服务攻防-端口协议桌面应用QQWPS 等 RCEhydra 口令猜解未授权检测

Day84&#xff1a;服务攻防-端口协议&桌面应用&QQ&WPS等RCE&hydra口令猜解&未授权检测_wps漏洞复现 rce-CSDN博客https://blog.csdn.net/qq_61553520/article/details/137119893?ops_request_misc%257B%2522request%255Fid%2522%253A%25220E34BCAF-166A-4…

《蓝桥杯算法入门》(C/C++、Java、Python三个版本)24年10月出版

推荐&#xff1a;《算法竞赛》&#xff0c;算法竞赛大全书&#xff0c;网购&#xff1a;京东 天猫  当当 文章目录 《蓝桥杯算法入门》内容简介本书读者对象作者简介联系与交流《蓝桥杯算法入门 C/C》版目录 《蓝桥杯算法入门 Java》版目录 《蓝桥杯算法入门 Python》版目录 …

scrapy爬取汽车、车评数据【上】

这个爬虫我想分三期来写&#xff1a; ✅ 第一期写如何爬取汽车的车型信息&#xff1b; ✅ 第二期写如何爬取汽车的车评&#xff1b; ✅ 第三期写如何对车评嵌入情感分析结果&#xff0c;以及用简单的方法把数据插入mysql中&#xff1b; 技术基于scrapy框架、BERT语言模型、mysq…