我正在寻找比下面的查询更有效和/或更容易阅读的内容。解释问题的最佳方法是提供样本。
假设MySQL中的以下表结构代表我正在创建的应用程序中各种字符串的本地化上下文。
create table LOCALIZATION_TABLE (
SET_NAME varchar(36) not null,
LOCALE varchar(8) not null default '_',
ENTRY_KEY varhcar(36) not null,
ENTRY_VALUE text null
);
alter table LOCALIZATION_TABLE
add constraint UQ_ENTRY
unique (SET_NAME, LOCALE, ENTRY_KEY);
假设在表格中输入以下值:
insert into LOCALIZATION_TABLE (SET_NAME, LOCALE, ENTRY_KEY, ENTRY_VALUE)
values
('STD_TEXT', '_', 'HELLO', 'Hello!'),
('STD_TEXT', '_', 'GOODBYE', 'Goodbye.'),
('STD_TEXT', 'ge', 'GOODBYE', 'Lebewohl')
;
我想选择德语(“ge”)的所有可用条目,如果不可用,则默认使用英文文本(“_”)。我目前使用的查询如下:
select * from LOCALIZATION_TABLE where SET_NAME = 'STD_TEXT' and LOCALE = 'ge'
union
select * from LOCALIZATION_TABLE where SET_NAME = 'STD_TEXT' and LOCALE = '_'
and ENTRY_KEY not in (
select ENTRY_KEY from LOCALIZATION_TABLE where BUNDLE = 'STD_TEXT' and LOCALE = 'ge'
)
我真的不喜欢这个查询的外观,我确信必须有一些更简洁的东西可以用来代替。任何帮助或正确方向的线索将不胜感激。虽然这样做但它似乎不是正确的。
答案 0 :(得分:3)
您可以提供自定义排序,然后选择第一行,如下所示:
select * from (
select *
from LOCALIZATION_TABLE
where SET_NAME = 'STD_TEXT'
order by field(LOCALE, 'ge', '_') -- this provides the custom ordering
) x
group by ENTRY_KEY; -- this captures the first row for each ENTRY_KEY
内部选择order by field(LOCALE, 'ge', '_')
以您定义的顺序获取行 - 在这种情况下,德语首先是否存在,然后是英语(您可以在列表中添加更多语言)。
这里的“技巧”是在不列出非组列时使用mysql的“非标准”GROUP BY
行为(大多数服务器将此视为语法错误) - 它只是返回为每个组找到的第一行。外部选择使用group by
而不使用聚合来获取每个命名组的第一行。
使用您的数据输出此查询:
+----------+--------+-----------+-------------+
| SET_NAME | LOCALE | ENTRY_KEY | ENTRY_VALUE |
+----------+--------+-----------+-------------+
| STD_TEXT | ge | GOODBYE | Lebewohl |
| STD_TEXT | _ | HELLO | Hello! |
+----------+--------+-----------+-------------+
答案 1 :(得分:1)
我认为您的查询很好。但这是另一种方法:
SELECT
en.SET_NAME
, COALESCE(ge.LOCALE, en.LOCALE)
, en.ENTRY_KEY
, COALESCE(ge.ENTRY_VALUE, en.ENTRY_VALUE)
FROM
LOCALIZATION_TABLE AS en
LEFT JOIN
LOCALIZATION_TABLE AS ge
ON ge.ENTRY_KEY = en.ENTRY_KEY
AND ge.LOCALE = 'ge'
AND ge.SET_NAME = 'STD_TEXT'
WHERE
en.LOCALE = '_'
AND en.SET_NAME = 'STD_TEXT'