如何在 PHP 中将关联查询结果按视频分组并合并标签

16次阅读

如何在 PHP 中将关联查询结果按视频分组并合并标签

本文介绍如何将 mysql 多表关联查询(视频-标签中间表)的结果,从“一行一标签”的扁平结构,重构为“一个视频对应多个标签”的嵌套数组结构,并在 html 表格中去重展示,避免视频信息重复渲染。

在使用 mysqli_fetch_assoc() 处理多对多关系(如视频与标签)时,原始 SQL 查询会为每个视频-标签组合返回一条记录,导致同一视频在结果集中多次出现。这虽便于逐行遍历,却不利于前端展示——我们通常希望每个视频只显示一次,其所有标签以逗号分隔或列表形式聚合呈现

解决该问题的核心思路是:php 层面对结果集进行逻辑分组与结构重组,而非依赖 SQL 聚合(因 GROUP_CONCAT 有长度限制且灵活性低)。以下是推荐的实现方案:

✅ 步骤一:执行查询并获取完整结果集

$query = "SELECT * FROM video_tags VT           INNER JOIN videos V ON V.id_video = VT.id_video            INNER JOIN tags T ON T.id_tag = VT.id_tag           ORDER BY VT.id_video";  $prepare = mysqli_prepare($connexion, $query); mysqli_stmt_execute($prepare); $result = mysqli_stmt_get_result($prepare);

⚠️ 注意:此处应使用 mysqli_stmt_get_result() 后配合 while ($row = mysqli_fetch_assoc($result)) 循环读取全部数据,而非仅调用一次 mysqli_fetch_assoc() 获取首行(原代码中 $item_row = mysqli_fetch_assoc($item) 只取了第一行,后续 do/while 才继续取,易出错且不直观)。

✅ 步骤二:构建分组嵌套数组(视频为主键,标签为子数组)

$videos = []; while ($row = mysqli_fetch_assoc($result)) {     $videoId = $row['id_video'];      // 若该视频尚未初始化,则创建主结构     if (!isset($videos[$videoId])) {         $videos[$videoId] = [             'id_video'          => $videoId,             'video_title'       => $row['video_title'],             'video_description' => $row['video_description'],             'video_url'         => $row['video_url'],             'tags'              => [] // 初始化空标签数组         ];     }      // 将当前标签追加到该视频的 tags 子数组中     $videos[$videoId]['tags'][] = [         'id_video_tags' => $row['id_video_tags'],         'id_tag'        => $row['id_tag'],         'tag'           => $row['tag']     ]; }

该结构最终生成类似以下的 PHP 数组:

$videos = [     143 => [         'id_video' => 143,         'video_title' => 'Intro to PHP',         'tags' => [             ['id_video_tags'=>435, 'id_tag'=>12, 'tag'=>'PHP'],             ['id_video_tags'=>503, 'id_tag'=>50, 'tag'=>'Tutorial']         ]     ],     // ... 其他视频 ];

✅ 步骤三:在 html 中安全渲染(去重 + 标签聚合)

                                              lspecialchars(implode(', ', array_column($video['tags'], 'tag')))                 : 'No tags'; ?>                

? 安全提示:始终对输出到 HTML 的变量使用 htmlspecialchars() 防止 xss 攻击,尤其当 video_title 或 tag 可能含用户输入内容时。

立即学习PHP免费学习笔记(深入)”;

? 补充说明与优化建议

  • 性能考量:该方案在 PHP 内存中完成分组,适用于中等规模数据(数千条记录以内)。若数据量极大,可考虑数据库层使用 GROUP_CONCAT(tag SEPARATOR ‘, ‘) 配合 GROUP BY id_video,但需注意 group_concat_max_len 配置限制。
  • 扩展性:如需支持标签编辑、删除,可保留 id_video_tags 和 id_tag 用于后续操作;array_column(…, ‘tag’) 仅用于展示,不破坏原始结构。
  • 错误处理:生产环境建议添加 mysqli_error($connexion) 检查查询失败,并对 $result 是否为 false 做判空处理。

通过这种“先取全量、再分组聚合”的方式,你既能保持 SQL 查询简洁清晰,又能灵活控制前端展示逻辑,是处理多对多关联数据的经典实践。

text=ZqhQzanResources