1

我在 oracle 数据库中有 3 个表。它们看起来像这样:

Table User:

username|forename|surname
a          a       a
b          b       b
c          c       c


Table Right:

username|organisationname|right
a           x              user
a           x              admin
a           x              owner
a           y              user
a           y              admin
a           z              owner
b           x              user
c           y              user
c           y              admin
c           z              user

Table Organisation:

organisationname|number
x                 12
y                 14
z                 42

该表Right有两个foreign_keys:username指向表中同名的列userorganisationname指向表Organisation

对于user在 x 上拥有正确用户的所有用户organisation,我想获取organisations用户拥有 1 个或多个权限的所有(组织 x 除外)的列表。

在我的示例中:a 在 x 上有正确的用户,b 在 x 上有正确的用户,c 在 x 上没有正确的用户。所以我只想为用户 a 和 b (而不是 c !!!)获取所有组织(不同的!,没有组织名称两次)。如果一个用户在 x 之外没有其他组织的权限(他必须拥有正确的用户),我希望该用户在输出中但没有组织。

预期的输出是:

user|organisations
a    y; z
b  

我希望你明白我的意思。到目前为止,我的查询如下:

select username, 
/* listagg put all the organisationnames of one user in one column separated by ;  
   then the organsation x gets removed from the result*/
regexp_replace (listagg (organisationname, '; ') 
       within group (order by organisationname), '(^x;?)|( ?x;)|(; x$) ', '') as "organisations"
from(
    select u2.username as username,
           o.organisationname as organisationname,
           /*I do this to remove duplicate entries in the list of organisations of one user */
           row_number() over (partition by u2.username, o.organisationname order by u2.username) as rn
       from RIGHT r2, ORGANISATION o, USER u2 
       where r2.organisationname = o.organisationname
       and r2.username = u2.username
       and r2.username IN
           (
            select u.username
            from USER u, right r
            where r.username = u.username
            and r.right = 'user'
            )  
     order by u2.username, o.organisationname         
)
where rn = 1
group by username

该查询有效,但现在我想过滤列组织。但我无法在 where 子句中使用它。当我尝试使用它时,oracle 会抛出错误:ORA-00904: invalid identifier

有谁知道我怎么能做到这一点?我认为oracle有问题是因为listagg的解析函数。我也会很高兴关于如何简化我的查询的建议。谢谢!

这是我尝试在 where 子句中使用列组织的内容:

select * from ( ***long query shown above***) 
where organisations like '%termToSearch%';  
4

2 回答 2

2

一步一步:用户|x的用户名,然后是除x之外的不同用户|组织,然后是每个用户名的聚合:

select
  username,
  listagg(organisationname, '; ') within group (order by organisationname) as organisations
from
(
  select distinct username, organisationname
  from right
  where username in 
  (
    select username 
    from right 
    where organisationname = 'x' and right = 'user'
  )
  and organisationname <> 'x'
)
group by username;

(不幸LISTAGG的是不接受DISTINCT关键字,所以我们需要两个步骤而不是一个步骤来构建不同组织的列表。)

更新:为了让用户也没有任何其他组织,我们将删除条件and organisationname <> 'x'并将 case 构造添加到LISTAGG

select
  username,
  listagg(case when organisationname <> 'x' then organisationname end, '; ')
    within group (order by organisationname) as organisations
from
(
  select distinct username, organisationname
  from right
  where username in 
  (
    select username 
    from right 
    where organisationname = 'x' and right = 'user'
  )
)
group by username;
于 2017-04-11T14:35:32.033 回答
1

更改要添加别名的代码部分,

旧代码 -

within group (order by organisationname), '(^x;?)|( ?x;)|(; x$) ', '') as "organisations"

新代码 -

 within group (order by organisationname), '(^x;?)|( ?x;)|(; x$) ', '') organisations

或者

新代码 -

within group (order by organizationname), '(^x;?)|( ?x;)|(; x$) ', '') as "ORGANISATIONS"

如果您仍想使用旧代码,您可以将最后一个 where 条件更改为 -

where "organisations" like '%termToSearch%'; 

参考:架构对象名称和限定符

于 2017-04-11T14:51:30.970 回答