Функция SQL Server для возврата значения или NULL, если нет результатов - PullRequest
0 голосов
/ 29 октября 2019

Я пытаюсь создать скалярную функцию, которая будет возвращать номер сотрудника, если вакансия ( код задания здесь) заполнена, или значение NULL, если оно вакантно ( A ctive или T аннулированный статус).

Data Preview

Я знаю, что должность менеджера (10061) вакантна в магазине 517. Та же самая позиция заполнена на 536. Я выполняю это с помощью CTE, который мешает мне использовать WHERE EXISTS, мою первую попытку решить эту проблему.

Самое близкое, что я получаю, это:

with cte as (
select store, empno, name, jobcode, jobtitle, status, lasthiredate, terminationdate, row_number() over (partition by jobcode, store order by terminationdate desc) as rn
from #temp
)
Select distinct a.store, b.empno, a.name, a.jobcode, a.jobtitle, a.status, a.rn
from cte a
left outer join cte b on a.jobcode = b.jobcode and a.store = b.store and a.status <> b.status
where a.jobcode = 10061
order by a.store, a.rn

А вот некоторые примеры данных, если вы хотите поиграть дома:

create table #temp (    [Store] int,    [EmpNo] int,    [Name] varchar(11),     [Jobcode] int,  [Dept] int,     [JobTitle] varchar(8),  [LastHireDate] date,    [Status] varchar(1),    [TerminationDate] date); 
Insert into #temp values (  119,    5042105,    'Gary D.',  10721,  10,     'Director',     '7/7/2003',     'T',    '1/18/2015'); 
Insert into #temp values (  119,    5391105,    'Sonia H.',     10721,  10,     'Director',     '12/19/2008',   'A',    NULL); 
Insert into #temp values (  119,    8155608,    'Paul W.',  10721,  10,     'Director',     '3/20/2017',    'T',    '11/30/2017'); 
Insert into #temp values (  119,    11952311,   'LARRY B.',     10721,  11,     'Director',     '4/15/2010',    'T',    '3/14/2012'); 
Insert into #temp values (  119,    19065019,   'Gary D.',  10721,  10,     'Director',     '7/7/2003',     'T',    '3/24/2017'); 
Insert into #temp values (  119,    19073019,   'Timothy P.',   10721,  10,     'Director',     '4/30/2013',    'T',    '12/5/2017'); 
Insert into #temp values (  119,    27230127,   'Jeffery F.',   10721,  10,     'Director',     '1/17/2010',    'T',    '12/21/2015'); 
Insert into #temp values (  119,    89113289,   'Timothy S.',   10721,  10,     'Director',     '8/3/2015',     'T',    '5/14/2019'); 
Insert into #temp values (  119,    89209289,   'Michael B.',   10721,  10,     'Director',     '12/17/2015',   'A',    NULL); 
Insert into #temp values (  119,    89453589,   'Harold H.',    10721,  10,     'Director',     '2/21/2018',    'T',    '5/7/2019'); 
Insert into #temp values (  119,    89604489,   'Jason B.',     10721,  10,     'Director',     '5/17/2017',    'A',    NULL); 
Insert into #temp values (  119,    89931089,   'Jeffery F.',   10721,  10,     'Director',     '1/17/2010',    'A',    NULL); 
Insert into #temp values (  119,    99371499,   'William A.',   10721,  10,     'Director',     '11/2/1998',    'A',    NULL); 
Insert into #temp values (  119,    99728099,   'K. Renee H.',  10721,  10,     'Director',     '9/11/1989',    'T',    '3/24/2017'); 
Insert into #temp values (  517,    11263511,   'Michael D.',   10061,  3,  'Manager',  '1/19/2015',    'T',    '7/27/2015'); 
Insert into #temp values (  517,    11544211,   'Richard L.',   10061,  3,  'Manager',  '10/10/2005',   'T',    '12/14/2014'); 
Insert into #temp values (  536,    3507003,    'Jeffrey S.',   10061,  3,  'Manager',  '2/18/2002',    'T',    '6/8/2012'); 
Insert into #temp values (  536,    12558412,   'John S.',  10061,  3,  'Manager',  '9/27/2010',    'A',    NULL); 

В идеале вывод функции будет

Store   |   Jobcode |   EmpNo
 536        10061       12558412
 517        10061       NULL

Любая помощьтак высоко ценится.

Ответы [ 2 ]

1 голос
/ 29 октября 2019

Чтобы получить информацию, которую вы ищете, вам просто нужно просмотреть запись с самыми последними LastHireDate для каждого store и jobcode. Если эта запись имеет статус 'A', позиция укомплектована, иначе она вакантна.

Я бы предложил использовать для фильтрации коррелированный подзапрос вместо row_number(). В этой ситуации это может быть эффективным решением, особенно с индексом (store, jobcode, LastHireDate).

select store, jobcode, case when status = 'A' then name end employeeName
from #temp t
where t.LastHireDate = (
    select max(t1.LastHireDate) 
    from #temp t1 
    where t1.store = t.store and t1.jobcode = t.jobcode
)
order by store, jobcode

Демонстрация по DB Fiddle :

store | jobcode | employeeName
----: | ------: | :-----------
  119 |   10721 | <em>null</em>        
  517 |   10061 | <em>null</em>        
  536 |   10061 | John S.     
0 голосов
/ 29 октября 2019

Вот скалярная функция для этого:

CREATE FUNCTION dbo.CheckAvailability (@jobcode INT, @store INT)
RETURNS INT
AS
BEGIN
DECLARE @return int
SELECT @return = t.jobcode
FROM temp1 AS t
WHERE t.Jobcode = @jobcode
  AND t.Store = @store
  AND t.Status NOT IN ('A','T')

RETURN (@return)
END
Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...