Đây là một cách hay để loại bỏ trùng lặp bản ghi trong bảng có cột nhận dạng dựa trên khóa chính mong muốn mà bạn có thể xác định trong thời gian chạy. Trước khi bắt đầu, tôi sẽ điền một tập dữ liệu mẫu để làm việc bằng cách sử dụng mã sau:
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
Tiếp theo, tôi sẽ tạo một Loại có tên là ColumnNames:
create type ColumnNames AS table
(Columnnames varchar(max))
Cuối cùng, tôi sẽ tạo một proc được lưu trữ với 3 lưu ý sau: 1. Proc sẽ nhận tham số bắt buộc @tablename xác định tên của bảng mà bạn đang xóa trong cơ sở dữ liệu của mình. 2. Proc có một tham số tùy chọn @columns mà bạn có thể sử dụng để xác định các trường tạo nên khóa chính mong muốn mà bạn đang xóa. Nếu trường này bị bỏ trống, giả sử rằng tất cả các trường bên cạnh cột nhận dạng tạo thành khóa chính mong muốn. 3. Khi các bản ghi trùng lặp bị xóa, bản ghi có giá trị thấp nhất trong cột nhận dạng của nó sẽ được duy trì.
Đây là proc đã lưu delete_dupes của tôi:
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
Khi điều này được tuân thủ, bạn có thể xóa tất cả các bản ghi trùng lặp của mình bằng cách chạy chương trình. Để xóa lỗi lừa đảo mà không xác định khóa chính mong muốn, hãy sử dụng lệnh gọi này:
exec delete_dupes '_original'
Để xóa bản sao dựa trên khóa chính mong muốn đã xác định, hãy sử dụng lệnh gọi này:
declare @table1 as columnnames
insert @table1
values ('salesrepid'),('sale')
exec delete_dupes '_original' , @table1