Удалить повторяющиеся записи в SQL Server?

Рассмотрим столбец с именем EmployeeName таблица Employee. Цель состоит в том, чтобы удалить повторяющиеся записи на основе поля EmployeeName.

EmployeeName
------------
Anand
Anand
Anil
Dipak
Anil
Dipak
Dipak
Anil

Используя один запрос, я хочу удалить повторяющиеся записи.

Как это можно сделать с помощью TSQL в SQL Server?


person usr021986    schedule 23.07.2010    source источник
comment
Вы имеете в виду удаление повторяющихся записей, верно?   -  person Sarfraz    schedule 23.07.2010
comment
вы могли бы выбрать отдельные значения и связанные с ними идентификаторы и удалить те записи, идентификаторы которых отсутствуют в уже выбранном списке?   -  person DaeMoohn    schedule 23.07.2010
comment
у вас есть уникальный столбец идентификатора?   -  person Andrew Bullock    schedule 23.07.2010
comment
Нет, у меня нет столбца Уникальный идентификатор   -  person usr021986    schedule 24.07.2010
comment
как вы приняли ответ Джона Гибба, если в таблице отсутствует уникальный идентификатор? где столбец empId в вашем примере используется Джоном?   -  person armen    schedule 02.10.2013
comment
Если у вас нет уникального столбца идентификатора или чего-либо еще, что имеет значение для упорядочения, вы МОЖЕТЕ также упорядочить по столбцу имени сотрудника... так что ваш rn будет row_number() over (partition by EmployeeName order by EmployeeName)... это выберет произвольную единственную запись для каждого имя.   -  person John Gibb    schedule 22.11.2013
comment
Возможный дубликат Как удалить повторяющиеся строки?   -  person Erik    schedule 05.12.2017


Ответы (8)


Вы можете сделать это с помощью оконных функций. Он упорядочит дубликаты по empId и удалит все, кроме первого.

delete x from (
  select *, rn=row_number() over (partition by EmployeeName order by empId)
  from Employee 
) x
where rn > 1;

Запустите его как select, чтобы увидеть, что будет удалено:

select *
from (
  select *, rn=row_number() over (partition by EmployeeName order by empId)
  from Employee 
) x
where rn > 1;
person John Gibb    schedule 23.07.2010
comment
Если у вас нет первичного ключа, вы можете использовать ORDER BY (SELECT NULL) stackoverflow.com/a/4812038. - person Arithmomaniac; 01.07.2016

Предполагая, что ваша таблица Employee также имеет уникальный столбец (ID в приведенном ниже примере), будет работать следующее:

delete from Employee 
where ID not in
(
    select min(ID)
    from Employee 
    group by EmployeeName 
);

Это оставит версию с самым низким идентификатором в таблице.

Изменить
Комментарий МакГайвера по состоянию на SQL 2012

MIN можно использовать со столбцами numeric, char, varchar, uniqueidentifier или datetime, но не с битовыми столбцами.

Для 2008 R2 и более ранних версий:

MIN можно использовать с числовыми столбцами, столбцами char, varchar или datetime, но не с битовыми столбцами (а также не работает с GUID)

Для 2008R2 вам нужно привести GUID к типу, поддерживаемому MIN, например.

delete from GuidEmployees
where CAST(ID AS binary(16)) not in
(
    select min(CAST(ID AS binary(16)))
    from GuidEmployees
    group by EmployeeName 
);

SqlFiddle для различных типов в Sql 2008

SqlFiddle для различных типов в Sql 2012

person StuartLC    schedule 23.07.2010
comment
Кроме того, в Oracle вы можете использовать rowid, если нет другого столбца с уникальным идентификатором. - person Brandon Horsley; 23.07.2010
comment
+1 Даже если бы не было столбца идентификатора, его можно было бы добавить в качестве поля идентификации. - person Kyle B.; 23.07.2010
comment
Отличный ответ. Острый и эффективный. Даже если у таблицы нет идентификатора; лучше включить один для выполнения этого метода. - person MiBol; 13.11.2018

Вы можете попробовать что-то вроде следующего:

delete T1
from MyTable T1, MyTable T2
where T1.dupField = T2.dupField
and T1.uniqueField > T2.uniqueField  

(предполагается, что у вас есть уникальное целочисленное поле)

Лично я бы сказал, что вам лучше попытаться исправить тот факт, что повторяющиеся записи добавляются в базу данных до того, как это произойдет, а не в качестве операции по исправлению.

person Ben Cawley    schedule 23.07.2010
comment
У меня нет уникального поля (ID) в моей таблице. Как я могу выполнить операцию тогда. - person usr021986; 24.07.2010

WITH CTE AS
(
   SELECT EmployeeName, 
          ROW_NUMBER() OVER(PARTITION BY EmployeeName ORDER BY EmployeeName) AS R
   FROM employee_table
)
DELETE CTE WHERE R > 1;

Магия общих табличных выражений.

person Mostafa Elmoghazi    schedule 25.07.2010
comment
SubPortal / a_horse_with_no_name — разве это не должен быть выбор из фактической таблицы? Кроме того, ROW_NUMBER должно быть ROW_NUMBER(), потому что это функция, верно? - person MacGyver; 17.02.2014

Пытаться

DELETE
FROM employee
WHERE rowid NOT IN (SELECT MAX(rowid) FROM employee
GROUP BY EmployeeName);
person Anurag Garg    schedule 02.10.2013

Если вы ищете способ удалить дубликаты, но у вас есть внешний ключ, указывающий на таблицу с дубликатами, вы можете воспользоваться следующим подходом, используя медленный, но эффективный курсор.

Он переместит повторяющиеся ключи в таблицу внешних ключей.

create table #properOlvChangeCodes(
    id int not null,
    name nvarchar(max) not null
)

DECLARE @name VARCHAR(MAX);
DECLARE @id INT;
DECLARE @newid INT;
DECLARE @oldid INT;

DECLARE OLVTRCCursor CURSOR FOR SELECT id, name FROM Sales_OrderLineVersionChangeReasonCode; 
OPEN OLVTRCCursor;
FETCH NEXT FROM OLVTRCCursor INTO @id, @name;
WHILE @@FETCH_STATUS = 0  
BEGIN  
        -- determine if it should be replaced (is already in temptable with name)
        if(exists(select * from #properOlvChangeCodes where Name=@name)) begin
            -- if it is, finds its id
            Select  top 1 @newid = id
            from    Sales_OrderLineVersionChangeReasonCode
            where   Name = @name

            -- replace terminationreasoncodeid in olv for the new terminationreasoncodeid
            update Sales_OrderLineVersion set ChangeReasonCodeId = @newid where ChangeReasonCodeId = @id

            -- delete the record from the terminationreasoncode
            delete from Sales_OrderLineVersionChangeReasonCode where Id = @id
        end else begin
            -- insert into temp table if new
            insert into #properOlvChangeCodes(Id, name)
            values(@id, @name)
        end

        FETCH NEXT FROM OLVTRCCursor INTO @id, @name;
END;
CLOSE OLVTRCCursor;
DEALLOCATE OLVTRCCursor;

drop table #properOlvChangeCodes
person Peter    schedule 28.09.2016

См. также приведенный ниже способ удаления.

Declare @Employee table (EmployeeName varchar(10))

Insert into @Employee values 
('Anand'),('Anand'),('Anil'),('Dipak'),
('Anil'),('Dipak'),('Dipak'),('Anil')

Select * from @Employee

введите здесь описание изображения

Создал образец таблицы с именем @Employee и загрузил ее с заданными данными.

Delete  aliasName from (
Select  *,
        ROW_NUMBER() over (Partition by EmployeeName order by EmployeeName) as rowNumber
From    @Employee) aliasName 
Where   rowNumber > 1

Select * from @Employee

Результат:

введите здесь описание изображения

Я знаю, что этот вопрос был задан шесть лет назад, и публикую на всякий случай, если кому-то это будет полезно.

person Jithin Shaji    schedule 28.09.2016

Вот хороший способ дедупликации записей в таблице, в которой есть столбец идентификаторов, основанный на желаемом первичном ключе, который вы можете определить во время выполнения. Прежде чем я начну, я заполню образец набора данных для работы, используя следующий код:

if exists (select 1 from sys.all_objects where type='u' and name='_original')
drop table _original

declare @startyear int = 2017
declare @endyear int = 2018
declare @iterator int = 1
declare @income money = cast((SELECT round(RAND()*(5000-4990)+4990 , 2)) as money)
declare @salesrepid int = cast(floor(rand()*(9100-9000)+9000) as varchar(4))
create table #original (rowid int identity, monthyear varchar(max), salesrepid int, sale money)
while @iterator<=50000 begin
insert #original 
select (Select cast(floor(rand()*(@endyear-@startyear)+@startyear) as varchar(4))+'-'+ cast(floor(rand()*(13-1)+1) as varchar(2)) ),  @salesrepid , @income
set  @salesrepid  = cast(floor(rand()*(9100-9000)+9000) as varchar(4))
set @income = cast((SELECT round(RAND()*(5000-4990)+4990 , 2)) as money)
set @iterator=@iterator+1
end  
update #original
set monthyear=replace(monthyear, '-', '-0') where  len(monthyear)=6

select * into _original from #original

Далее я создам тип с именем ColumnNames:

create type ColumnNames AS table   
(Columnnames varchar(max))

Наконец, я создам хранимую процедуру со следующими тремя оговорками: 1. Процедура будет принимать обязательный параметр @tablename, который определяет имя таблицы, которую вы удаляете из своей базы данных. 2. В процедуре есть необязательный параметр @columns, который вы можете использовать для определения полей, составляющих желаемый первичный ключ, который вы удаляете. Если это поле оставить пустым, предполагается, что все поля, кроме столбца идентификаторов, составляют желаемый первичный ключ. 3. При удалении повторяющихся записей будет сохранена запись с наименьшим значением в ее столбце идентификации.

Вот моя хранимая процедура delete_dupes:

 create proc delete_dupes (@tablename varchar(max), @columns columnnames readonly) 
 as
 begin

declare @table table (iterator int, name varchar(max), is_identity int)
declare @tablepartition table (idx int identity, type varchar(max), value varchar(max))
declare @partitionby varchar(max)  
declare @iterator int= 1 


if exists (select 1 from @columns)  begin
declare @columns1 table (iterator int, columnnames varchar(max))
insert @columns1
select 1, columnnames from @columns
set @partitionby = (select distinct 
                substring((Select ', '+t1.columnnames 
                From @columns1 t1
                Where T1.iterator = T2.iterator
                ORDER BY T1.iterator
                For XML PATH ('')),2, 1000)  partition
From @columns1 T2 )

end

insert @table 
select 1, a.name, is_identity from sys.all_columns a join sys.all_objects b on a.object_id=b.object_id
where b.name = @tablename  

declare @identity varchar(max)= (select name from @table where is_identity=1)

while @iterator>=0 begin 
insert @tablepartition
Select          distinct case when @iterator=1 then 'order by' else 'over (partition by' end , 
                substring((Select ', '+t1.name 
                From @table t1
                Where T1.iterator = T2.iterator and is_identity=@iterator
                ORDER BY T1.iterator
                For XML PATH ('')),2, 5000)  partition
From @table T2
set @iterator=@iterator-1
end 

declare @originalpartition varchar(max)

if @partitionby is null begin
select @originalpartition  = replace(b.value+','+a.type+a.value ,'over (partition by','')  from @tablepartition a cross join @tablepartition b where a.idx=2 and b.idx=1
select @partitionby = a.type+a.value+' '+b.type+a.value+','+b.value+') rownum' from @tablepartition a cross join @tablepartition b where a.idx=2 and b.idx=1
 end
 else
 begin
 select @originalpartition=b.value +','+ @partitionby from @tablepartition a cross join @tablepartition b where a.idx=2 and b.idx=1
 set @partitionby = (select 'OVER (partition by'+ @partitionby  + ' ORDER BY'+ @partitionby + ','+b.value +') rownum'
 from @tablepartition a cross join @tablepartition b where a.idx=2 and b.idx=1)
 end


exec('select row_number() ' + @partitionby +', '+@originalpartition+' into ##temp from '+ @tablename+'')


exec(
'delete a from _original a 
left join ##temp b on a.'+@identity+'=b.'+@identity+' and rownum=1  
where b.rownum is null')

drop table ##temp

end

Как только это будет выполнено, вы можете удалить все свои повторяющиеся записи, запустив proc. Чтобы удалить дубликаты без определения желаемого первичного ключа, используйте этот вызов:

exec delete_dupes '_original'

Чтобы удалить дубликаты на основе определенного желаемого первичного ключа, используйте этот вызов:

declare @table1 as columnnames
insert @table1
values ('salesrepid'),('sale')
exec delete_dupes '_original' , @table1
person Daniel Marcus    schedule 13.03.2018