select * from table where column = something,或者,当不可用时,column =其他

时间:2011-09-19 15:23:48

标签: mysql join views union

我正在寻找比下面的查询更有效和/或更容易阅读的内容。解释问题的最佳方法是提供样本。

假设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'
  )

真的不喜欢这个查询的外观,我确信必须有一些更简洁的东西可以用来代替。任何帮助或正确方向的线索将不胜感激。虽然这样做但它似乎不是正确的

2 个答案:

答案 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'
相关问题