0

我想编写以下查询,以便从工作表中返回所有工作,其中专业和子专业对等于某个用户的专业和子专业对之一(4 对之一)。如果用户有一对子专业 = 0,这意味着获取相应专业的所有子专业:

像这样的东西:

select * from jobs j
where 
      (j.profession, j.subprofession) in 

      (select (u.profession1, u.subprofession1) from users u where userid=@userid),
       select (u.profession2, u.subprofession2) from users u where userid=@userid),
       select (u.profession3, u.subprofession3) from users u where userid=@userid),
       select (u.profession4, u.subprofession4) from users u where userid=@userid)
and
j.profession in (select u.profession1, u.profession2, u.profession3, u.profession4 from users u where userid=@userid) and (u.subprofession1 = 0 or u.subprofession2 = 0 or u.subprofession3 = 0 or u.subprofession4 = 0))

我知道这个查询在语法上是错误的,并且没有做它应该做的事情,那么使用 IN 子句或 WHERE EXISTS 实现上述目的的方法是什么?

jobs: profession, subprofession 
1      (100,        200)
2      (100,        300)
3      (100,        400)
4      (100,        500)
5      (200,        300)
6      (400,        500)
7      (400,        100)
8      (400,        600)
9      (200,        200)
10     (600,        700)
11     (100,        100)
12     (500,        300)
13     (200,        200)

users: (prof1, subprof1, prof2, subprof2, prof3, subprof3, prof4, subprof4)
1       (100,   100,      757,   646,      100,   0,        500,   400)
2       (100,   2,        565,   76,       567,   534,      433,   565)
3       (200,   454,      553,   345,      354,   435,      334,   877)
4       (500,   300,      456,   565,      354,   435,      545,   435)
5       (400,   453,      434,   453,      423,   234,      324,   4435)
6       (100,   400,      435,   543,      465,   654,      454,   543)
7       (435,   435,      600,   700,      100,   0,        500,   400)
8       (100,   100,      553,   345,      100,   0,        500,   400)

请求的查询将从作业表行返回:1,2,3,4,11

这些是职业和子职业的值(作为一对)出现在用户 prof 和 subprof 对之一中的行。另外,因为用户 1 有一个条目(prof3=100 和 subprof3=0),查询应该返回 proession 为 100 的所有订单(对于所有子专业)。

4

2 回答 2

1

这应该适合你:

SELECT  *
FROM    Jobs
WHERE   EXISTS
        (   SELECT  1
            FROM    Users
            WHERE   Users.UserID = @UserID
            AND (   (Users.Profession1 = Jobs.Profession AND Users.SubProfession1 IN (Jobs.SubProfession, 0))
                OR  (Users.Profession2 = Jobs.Profession AND Users.SubProfession2 IN (Jobs.SubProfession, 0))
                OR  (Users.Profession3 = Jobs.Profession AND Users.SubProfession3 IN (Jobs.SubProfession, 0))
                OR  (Users.Profession4 = Jobs.Profession AND Users.SubProfession4 IN (Jobs.SubProfession, 0))
                )
        );

SQL Fiddle 示例

编辑

由于它是 SQL-Server,您可以使用CROSS APPLY..VALUESunpivot 数据,这样您可以内部连接并确定作业是否匹配,因为 subprofession 为 0,或者是否存在精确的 subprofession 匹配:

WITH UserProf AS
(   SELECT  DISTINCT
            Users.UserID,
            p.Profession,
            p.SubProfession
    FROM    Users
            CROSS APPLY
            (VALUES
                (Profession1, SubProfession1),
                (Profession2, SubProfession2),
                (Profession3, SubProfession3),
                (Profession4, SubProfession4)
            ) p (Profession, SubProfession)
    WHERE   Users.UserID = @UserID
)
SELECT  Jobs.*, 
        MatchType = CASE WHEN MIN(UserProf.SubProfession) = 0 THEN 'All Subprofession' ELSE 'Exact subprofession' END
FROM    Jobs
        INNER JOIN UserProf
            ON UserProf.Profession = Jobs.Profession
            AND UserProf.SubProfession IN (0, Jobs.SubProfession)
GROUP BY Jobs.JobID, Jobs.Profession, Jobs.SubProfession;

SQL-Fiddle 示例

于 2013-05-30T11:50:24.987 回答
0

我更愿意看两年前写的一个自我记录的查询,而不是我必须弄清楚它在做什么的查询。当然,对我来说令人困惑的事情可能是别人的简单。

鉴于子专业插槽中的通配符功率为零,意思是“专业匹配,不要担心子专业”,我会将查询作为两组的并集来处理。

Set 1 是与用户的职业匹配的所有工作的集合,其中 subprofession = 0。
Set 2 是与用户的职业/副职业二元组匹配的所有工作的集合。

因此,第一步是为每个用户获取子专业无关紧要的不同专业的列表。这与专业工作有内在联系。

第二步是为每个用户获取不同的二元组列表,即子专业在哪里很重要。这与专业和子专业的工作有内在联系。

如果您愿意,每个连接都可以附加一个指示 MatchLevel 的列,即第一组中的 ProfessionOnly 和第二组中的 Both,但是您可以看到相同的工作列出了两次,一次用于一般匹配,一次用于特定匹配。

于 2013-05-30T13:12:43.017 回答