SQL连接导致结果重复。

4
我按照以下方式创建了表格:

create table utilisateur(
 id_util number(10) primary key,
 nom varchar2(10) not null,
 prenom varchar2(10) not null,
 date_naissance date not null,
 adress varchar2(20)
);

create table cour(
 id_cour number(10) primary key,
 c_nom varchar2(20) not null,
 auteur varchar2(20) not null
);

create table etude(
 fk_util number(10) references utilisateur(id_util),
 fk_cour number(10) references cour(id_cour),
 primary key(fk_util,fk_cour)
);

create table examen(
 id_ex number(10) primary key,
 ex_nom varchar2(20) not null,
 temp date,
 fk_cour number(10) references cour(id_cour)
);

create table passer(
 fk_util number(10) references utilisateur(id_util),
 fk_ex number(10) references examen(id_ex),
 primary key(fk_util,fk_ex),
 note number(4)
);

create table certificat(
 cert_nom varchar2(20),
 prix varchar2(10),
 code varchar2(10) primary key,
 fk_ex number(10),
 fk_util number(10)
);

create table signet(
 id_sign number(10) primary key,
 s_nom varchar2(20) not null,
 depand_par varchar2(20) not null,
 fk_util number(10) references utilisateur(id_util)
);

问题是我想查看所有用户(utilisateur)正在阅读哪些课程(cour),他们通过了哪些考试(examen)以及获得了哪些证书(certificat)。
我尝试使用内部连接、左连接、右连接、全连接和视图来实现这一点,但都没有成功。如果我有3个已注册的课程和2个考试,则会看到某些重复的内容。我认为可能是我的数据库出了问题。

enter image description here


1
你可以发布一下你正在使用的查询吗?这两个考试是针对同一门课程吗?在这个模式下是可能的,并且可能是导致你描述的行为的原因。 - steve godfrey
这是查询语句:select utilisateur.nom, cour.c_nom, examen.ex_nom from utilisateur left join etude on utilisateur.id_util=etude.fk_util left join cour on etude.fk_cour=cour.id_cour left join passer on utilisateur.id_util=passer.fk_util left join examen on passer.fk_ex=examen.id_ex; - Morar Mihai
1个回答

1

这个查询

select utilisateur.nom
       , cour.c_nom
       , examen.ex_nom
from utilisateur left join etude on utilisateur.id_util=etude.fk_util 
left join cour on etude.fk_cour=cour.id_cour 
left join passer on utilisateur.id_util=passer.fk_util 
left join examen on passer.fk_ex=examen.id_ex;

没有考虑到考试与课程相关的事实。我认为针对您想要实现的目标,不必进行外连接,因此请尝试这个。

select utilisateur.nom
     , cour.c_nom
     , examen.ex_nom
from utilisateur join etude on utilisateur.id_util=etude.fk_util 
                 join cour on etude.fk_cour=cour.id_cour 
                 join passer on utilisateur.id_util=passer.fk_util 
                 join examen on cour.id_cour = examen.fk_cour;

编辑:这比我最初想象的要复杂一些,我已经添加了一个新的解决方案。首先,我给查询添加了一些别名,如果在某些版本的Oracle上不使用某些表的别名进行ansi连接,则会出现问题。由于别名通常是一件好事,我已经添加了它们。我还将一些表移动到内联视图中,以使问题更清晰。除了这个整理之外,我唯一真正的改变是添加了这一行:

AND paex.id_ex = coex.id_ex

我已经根据您描述的数据测试了这个查询,并且似乎可以实现您想要的功能。

    SELECT ut.id_util
  ,ut.nom
  ,coex.c_nom
  ,paex.id_ex
  ,paex.ex_nom FROM   utilisateur ut LEFT JOIN (SELECT c_nom
             ,ex_nom
             ,co.id_cour
             ,id_ex 
             ,et.fk_util               
       FROM etude et JOIN cour co ON et.fk_cour = co.id_cour
                LEFT JOIN examen ex2 ON co.id_cour = ex2.fk_cour) coex ON coex.fk_util = ut.id_util LEFT JOIN (SELECT *
       FROM passer pa JOIN examen ex  ON pa.fk_ex   = ex.id_ex) paex ON paex.fk_util = ut.id_util 
                                                                    AND paex.id_ex = coex.id_ex ORDER BY id_util;

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接