本文介绍如何将 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']
]
],
// ... 其他视频
];
= htmlspecialchars($video['id_video']); ?>
= htmlspecialchars($video['video_title']); ?>
= !empty($video['tags'])
? htmlspecialchars(implode(', ', array_column($video['tags'], 'tag')))
: 'No tags'; ?>
? 安全提示:始终对输出到 HTML 的变量使用 htmlspecialchars() 防止 XSS 攻击,尤其当 video_title 或 tag 可能含用户输入内容时。
立即学习“PHP免费学习笔记(深入)”;
通过这种“先取全量、再分组聚合”的方式,你既能保持 SQL 查询简洁清晰,又能灵活控制前端展示逻辑,是处理多对多关联数据的经典实践。