web-dev-qa-db-fra.com

Comment amener les employés avec leurs managers

Voici à quoi je veux que la sortie ressemble:

Employee   Emp# Manager Mgr# 
   BLAKE   7698 KING    7839 
   CLARK   7782 KING    7839 
   JONES   7566 KING    7839 
   MARTIN  7654 BLAKE   7698 
   ALLEN   7499 BLAKE   7698 
   TURNER  7844 BLAKE   7698 
   JAMES   7900 BLAKE   7698 
   WARD    7521 BLAKE   7698 
   FORD    7902 JONES   7566 
   SMITH   7369 FORD    7902 
   SCOTT   7788 JONES   7566 
   ADAMS   7876 SCOTT   7788 
   MILLER  7934 CLARK   7782 

Voici ce que j'ai obtenu:

 SQL> SELECT ename, empno, (SELECT ename FROM EMP WHERE empno = mgr)AS MANAGER, mgr from emp order by empno;

 ENAME           EMPNO MANAGER           MGR
 ---------- ---------- ---------- ----------
 SMITH            7369                  7902
 ALLEN            7499                  7698
 WARD             7521                  7698
 JONES            7566                  7839
 MARTIN           7654                  7698
 BLAKE            7698                  7839
 CLARK            7782                  7839
 SCOTT            7788                  7566
 KING             7839
 TURNER           7844                  7698
 ADAMS            7876                  7788

 ENAME           EMPNO MANAGER           MGR
 ---------- ---------- ---------- ----------
 JAMES            7900                  7698
 FORD             7902                  7566
 MILLER           7934                  7782

Je ne trouve pas pourquoi le champ gestionnaire est vide.

Voici le tableau:

 SQL> select empno, ename, job,deptno, mgr from emp;

 EMPNO ENAME      JOB           DEPTNO        MGR
 ---------- ---------- --------- ---------- ----------
  7839 KING       PRESIDENT         10
  7698 BLAKE      MANAGER           30       7839
  7782 CLARK      MANAGER           10       7839
  7566 JONES      MANAGER           20       7839
  7654 MARTIN     SALESMAN          30       7698
  7499 ALLEN      SALESMAN          30       7698
  7844 TURNER     SALESMAN          30       7698
  7900 JAMES      CLERK             30       7698
  7521 WARD       SALESMAN          30       7698
  7902 FORD       ANALYST           20       7566
  7369 SMITH      CLERK             20       7902

 EMPNO ENAME      JOB           DEPTNO        MGR
 ---------- ---------- --------- ---------- ----------
  7788 SCOTT      ANALYST           20       7566
  7876 ADAMS      CLERK             20       7788
  7934 MILLER     CLERK             10       7782

14 lignes sélectionnées.

10
user770022

Il s'agit d'une auto-jointure classique, essayez ce qui suit:

SELECT e.ename, e.empno, m.ename as manager, e.mgr
FROM
    emp e, emp m
WHERE e.mgr = m.empno

Et si vous souhaitez inclure le président qui n'a pas de gestionnaire, au lieu d'une jointure intérieure, utilisez une jointure extérieure dans la syntaxe Oracle:

SELECT e.ename, e.empno, m.ename as manager, e.mgr
FROM
    emp e, emp m
WHERE e.mgr = m.empno(+)

Ou dans la syntaxe SQL ANSI:

SELECT e.ename, e.empno, m.ename as manager, e.mgr
FROM
    emp e
    LEFT OUTER JOIN emp m
        ON e.mgr = m.empno
31
Xint0
(SELECT ename FROM EMP WHERE empno = mgr)

Il n'y a aucun enregistrement dans EMP qui répond à ces critères.

Vous devez vous auto-joindre pour obtenir cette relation.

SELECT e.ename AS Employee, e.empno, m.ename AS Manager, m.empno
FROM EMP AS e LEFT OUTER JOIN EMP AS m
ON e.mgr =m.empno;

MODIFIER:

La réponse que vous avez sélectionnée ne répertoriera pas votre président car il s'agit d'une jointure interne . Je pense que vous serez de retour lorsque vous découvrirez que votre sortie n'est pas ce que votre devoir (je soupçonne) de devoirs a exigé. Voici le cas de test réel:

> select * from emp;

 empno | ename |    job    | deptno | mgr  
-------+-------+-----------+--------+------
  7839 | king  | president |     10 |     
  7698 | blake | manager   |     30 | 7839
(2 rows)

> SELECT e.ename employee, e.empno, m.ename manager, m.empno
FROM emp AS e LEFT OUTER JOIN emp AS m
ON e.mgr =m.empno;

 employee | empno | manager | empno 
----------+-------+---------+-------
 king     |  7839 |         |      
 blake    |  7698 | king    |  7839
(2 rows)

La différence est qu'une jointure externe renvoie toutes les lignes. Une jointure interne produira les éléments suivants:

> SELECT e.ename, e.empno, m.ename as manager, e.mgr
FROM emp e, emp m
WHERE e.mgr = m.empno;

 ename | empno | manager | mgr  
-------+-------+---------+------
 blake |  7698 | king    | 7839
(1 row)
5
Brian Roach

ESSAYE ÇA

SELECT E.ename,E.empno,ISNULL(E.ename,'NO MANAGER') AS MANAGER FROM emp e
INNER JOIN emp M
ON  M.empno=E.empno

Au lieu de la sous-requête, utilisez l'auto-jointure

3
Pramod Deshmukh

Vous auriez pu changer votre requête en:

SELECT ename, empno, (SELECT ename FROM EMP WHERE empno = e.mgr)AS MANAGER, mgr 
from emp e 
order by empno;

Cela indiquerait au moteur que pour la table emp interne, empno doit être associé à la colonne mgr de la table externe. enter image description here

3
Lambs

Peut-être que votre sous-requête (SELECT ename FROM EMP WHERE empno = mgr) pense, donnez-moi les dossiers des employés qui sont leurs propres gestionnaires! (c'est-à-dire où l'empno d'une ligne est le même que le mgr de la ligne même.)

avez-vous envisagé de réécrire ceci pour utiliser une (auto) jointure interne? (Je demande, car je ne suis même pas sûr que ce qui suit fonctionnera ou non.)

SELECT t1.ename, t1.empno, t2.ename as MANAGER, t1.mgr
from emp as t1
inner join emp t2 ON t1.mgr = t2.empno
order by t1.empno;
2
Funka