45

我有一个博客帖子的标签字段。标签具有唯一 ID,但其 displayName 可能重复。我想要的是一个选择帖子的查询,在all_tags字段中我们得到几个 (id,displayName) 是这样的:

id1,name1;id2,name2;id3,name3

我的查询看起来像:

select ....
CONCAT_WS(';', DISTINCT (CONCAT_WS(',',tags.id,tags.displayName))) AS all_tags
Join ...post content ...
Join ...post_tags ...
Join ...tags ...
ORDER BY posts.id

此行会导致问题:

CONCAT_WS(';', DISTINCT (CONCAT_WS(',',tags.id,tags.displayName))) AS all_tags

我应该如何修改它?

有些人使用内部(SELECT .. FROM),但正如我所听说的,它是如此低效


SELECT `posts`.*,`categories`.*,`creators`.*,`editors`.*
CONCAT_WS(';', DISTINCT GROUP_CONCAT(CONCAT_WS(',',tags.id,tags.displayName))) AS all_ids
FROM (`posts`) 
LEFT JOIN `languages` ON `posts`.`language_id`=`languages`.`id` 
LEFT JOIN `users` as creators ON `posts`.`creatorUser_id`=`creators`.`id` 
LEFT JOIN `users` as editors ON `posts`.`lastEditorUser_id`=`editors`.`id` 
LEFT JOIN `userProfiles` as editors_profile ON `editors`.`profile_id`=`editors_profile`.`id` 
LEFT JOIN `categories` ON `posts`.`category_id`=`categories`.`id` 
LEFT JOIN `postTags` ON `postTags`.`post_id`=`posts`.`id` 
LEFT JOIN `tags` ON `postTags`.`tag_id`=`tags`.`id` 
LEFT JOIN `postTags` as `nodetag_checks` ON `nodetag_checks`.`post_id`=`posts`.`id` 
LEFT JOIN `tags` as `tag_checks` ON `nodetag_checks`.`tag_id`=`tag_checks`.`id` 
WHERE ( 9 IN(`tag_checks`.`id`,`tag_checks`.`cached_parents`) OR 10 IN(`tag_checks`.`id`,`tag_checks`.`cached_parents`) OR 11 IN(`tag_checks`.`id`,`tag_checks`.`cached_parents`)) 
GROUP BY `posts`.`id` ORDER BY `posts`.`created` desc LIMIT 0, 20  
4

3 回答 3

100

尝试这个:

GROUP_CONCAT(
  DISTINCT CONCAT(tags.id,',',tags.displayName) 
  ORDER BY posts.id 
  SEPARATOR ';'
)
于 2013-09-17T08:48:20.643 回答
3

根据@Willa 的建议,我将我的评论添加为 anwser :

GROUP_CONCAT 允许您连接多个字段:

GROUP_CONCAT(tags.id, ',', tags.displayName)

与斯蒂芬的答案的唯一区别是,如果您的代码允许同一标签多次影响一个帖子,或者如果您 JOIN 序列导致您在标签表中多选同一记录。在这种情况下,我的解决方案将多次返回相同的标签。

于 2020-02-16T10:42:40.747 回答
0

@Stephan的最佳答案之上,为了防止由于查询中的多个 JOIN 而多次显示相同的内容,但您不希望 id 显示在输出中......

GROUP_CONCAT(
    DISTINCT
    tags.displayName,
    '||__', tags.id, '__||'
    SEPARATOR '\n'
)

然后遍历结果,最后删除 ||__ 和 __|| 之间的所有内容 .

此示例适用于 php:

$data = preg_replace("/\|\|__.*__\|\|/", '',  $data);
于 2022-01-26T20:31:33.993 回答