1

我目前正在为 MS-Access 数据库编写一个导出函数,但我不太确定如何编写一个查询来提供我想要的结果。
我想做的是以下几点:

假设我有一张桌子Error,并且与 table 存在多对多关系,由 tableCause建模ErrorCause。目前我有一个与此类似的查询(简化,原件也进一步发展了一种关系):

select Error.ID, Cause.ID
from ((Error inner join ErrorCauses on Error.ID = ErrorCauses.Error)
      left join Cause on ErrorCauses.Cause = Cause.ID)

我得到这样的东西:

错误 | 原因
 -------------
 12345 | 12    
 12345 | 23    
 67890 | 23    
 67890 | 34    

但是我需要为每个错误选择第一个的 ID,比如 3 个原因(即使它们是空的),这样它看起来像这样:

错误 | 原因1 | 原因2 | 原因3
 --------------------------------
 12345 | 12 | 23 |       
 67890 | 23 | 34 |       

有没有办法在一个查询中做到这一点?喜欢选择前 3 名,然后将其展平到结果行中?在此先感谢您的任何指点。

4

1 回答 1

1

您的要求是针对特定数量的原因--3。这使得通过在同一个子查询上执行三向联接,在同一行上获得三个不同的原因成为可能并且易于管理。

首先,让我们将您的错误和原因查询定义为直接的 Access 查询(如果您想要技术的话,可以使用 QueryDef 对象)。

qryErrorCauseInfo

select
  Error.ID as ErrorID
, Cause.ID as CauseID
from (Error
inner join ErrorCauses
on Error.ID = ErrorCauses.Error)
left outer join Cause
on ErrorCauses.Cause = Cause.ID

顺便说一句,我觉得上面的左连接应该是一个内连接,原因我在评论中提到过。

接下来,让我们进行三向连接,以获取行中可能的原因组合:

qryTotalCause

select distinct
  *
, iif(Cause1 is null, 0, 1)
+ iif(Cause2 is null, 0, 1)
+ iif(Cause3 is null, 0, 1) as TotalCause
from (
  select
    eci1.ErrorID
  , eci1.CauseID as Cause1
  , iif(eci2.CauseID = Cause1, null, eci2.CauseID) as Cause2
  , iif(
      eci3.CauseID = Cause1 or eci3.CauseID = Cause2
    , null
    , eci3.CauseID
    ) as Cause3
  from (qryErrorCauseInfo as eci1
  left outer join qryErrorCauseInfo as eci2
  on eci1.ErrorID = eci2.ErrorID)
  left outer join qryErrorCauseInfo as eci3
  on eci2.ErrorID = eci3.ErrorID
) as sq
where (
  Cause1 < Cause2
  and Cause2 < Cause3
) or (
  Cause1 < Cause2
  and Cause3 is null
) or (
  Cause2 is null
  and Cause3 is null
) or (
  Cause1 is null
  and Cause2 is null
  and Cause3 is null
)

最后,我们需要一个相关子查询来为每个错误选择原因数量最多的一行(其余行只是相同原因的不同排列):

select
  ErrorID
, Cause1
, Cause2
, Cause3
from qryTotalCause as tc1
where tc1.TotalCause = (
  select max(tc2.TotalCause)
  from qryTotalCause as tc2
  where tc1.ErrorID = tc2.ErrorID
)

简单的!(不是 :-)

于 2013-09-28T03:00:03.397 回答