এসকিউএল সার্ভারে নকল রেকর্ড মুছবেন?


95

EmployeeNameসারণী নামের একটি কলাম বিবেচনা করুন Employee। লক্ষ্যটি EmployeeNameক্ষেত্রের উপর ভিত্তি করে পুনরাবৃত্তি রেকর্ডগুলি মুছে ফেলা হয় ।

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

একটি কোয়েরি ব্যবহার করে, আমি পুনরাবৃত্তি হওয়া রেকর্ডগুলি মুছতে চাই।

এসকিউএল সার্ভারে এটি কীভাবে টিএসকিউএল দিয়ে করা যায়?


আপনি ডুপ্লিকেট রেকর্ড মুছতে চান তাই না?
সরফরাজ

আপনি স্বতন্ত্র মান এবং তাদের সম্পর্কিত আইডিগুলি নির্বাচন করতে এবং যাদের আইডি ইতিমধ্যে নির্বাচিত তালিকায় নেই, সেই সমস্ত রেকর্ড মুছে ফেলতে পারবেন?
দায়েমুহন

4
আপনার কি একটি অনন্য আইডি কলাম আছে?
অ্যান্ড্রু বুলক

4
জন গিবের দেওয়া উত্তরটি আপনি কীভাবে গ্রহণ করবেন, যদি টেবিলটিতে অনন্য আইডির অভাব থাকে? empIdজন আপনার উদাহরণে কলামটি কোথায় ব্যবহার করেছেন?
আর্মেন

4
যদি আপনার কাছে কোনও অনন্য আইডি কলাম না থাকে বা অর্ডার করার অর্থপূর্ণ অন্য কোনও কিছু না থাকে তবে আপনি কর্মচারী কলামের মাধ্যমেও অর্ডার করতে পারেন ... সুতরাং আপনার আরএন হবে row_number() over (partition by EmployeeName order by EmployeeName)... এটি প্রতিটি নামের জন্য একটি স্বতন্ত্র একক রেকর্ড চয়ন করবে ।
জন গিব

উত্তর:


229

উইন্ডো ফাংশন দিয়ে আপনি এটি করতে পারেন। এটি এমপিআইপি দ্বারা দুপগুলি অর্ডার করবে এবং প্রথমটি বাদ দিয়ে সমস্ত মুছবে।

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

কী মুছে ফেলা হবে তা দেখার জন্য এটি নির্বাচন হিসাবে চালান:

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

4
আপনার যদি প্রাথমিক কী না থাকে তবে আপনি ORDER BY (SELECT NULL)
স্ট্যাকওভারফ্লো.

36

ধরে নিই যে আপনার কর্মচারী সারণিতেও একটি অনন্য কলাম রয়েছে ( IDনীচের উদাহরণে), নিম্নলিখিতগুলি কাজ করবে:

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

এটি সারণীতে সর্বনিম্ন আইডি সহ সংস্করণটি ছেড়ে যাবে।


ম্যাকগাইভারের মন্তব্য সম্পাদনা করুন - এসকিউএল ২০১২ অনুসারে

MIN সংখ্যাসূচক, চর, ভারচার, ইউনিক আইডেন্টিফায়ার বা ডেটটাইম কলামগুলির সাথে ব্যবহার করা যেতে পারে তবে বিট কলামগুলির সাথে নয়

জন্য 2008 R2 হলো এবং তার আগে,

এমআইএন সংখ্যা, চর, বারচার বা ডেটটাইম কলামগুলির সাথে ব্যবহার করা যেতে পারে তবে বিট কলামগুলির সাথে নয় (এবং এটি জিইউইডি এর সাথেও কাজ করে না)

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 
);

এসকিএল ২০০৮ সালে বিভিন্ন ধরণের জন্য এসকিএলফিডাল

এসকিএল 2012 এ বিভিন্ন ধরণের জন্য এসকিএলফিডাল


এছাড়াও, ওরাকলে, অন্য কোনও অনন্য আইডি কলাম না থাকলে আপনি "রোউইড" ব্যবহার করতে পারেন।
ব্র্যান্ডন হার্সলে

+1 আইডি কলাম না থাকলেও একটি পরিচয় ক্ষেত্র হিসাবে যুক্ত করা যেতে পারে।
কাইল বি।

দুর্দান্ত উত্তর। তীক্ষ্ণ এবং কার্যকর। এমনকি যদি টেবিলটির কোনও আইডি না থাকে; এই পদ্ধতিটি কার্যকর করতে একজনকে অন্তর্ভুক্ত করা ভাল।
MiBol

8

আপনি নিম্নলিখিত মত কিছু চেষ্টা করতে পারেন:

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

(এটি ধরে নেওয়া হয় যে আপনার একটি পূর্ণসংখ্য ভিত্তিক অনন্য ক্ষেত্র রয়েছে)

ব্যক্তিগতভাবে যদিও আমি বলব যে ডুপ্লিকেট এন্ট্রি পোস্ট ফিক্স-এর অপারেশন হিসাবে না হওয়ার আগে ডাটাবেসে ডুপ্লিকেট এন্ট্রি যুক্ত করা হচ্ছে সেই বিষয়টি সংশোধন করার চেষ্টা করা ভাল ছিল better


আমার সারণীতে অনন্য ক্ষেত্র (আইডি) নেই। তখন আমি কীভাবে অপারেশন করতে পারি।
usr021986

3
DELETE
FROM MyTable
WHERE ID NOT IN (
     SELECT MAX(ID)
     FROM MyTable
     GROUP BY DuplicateColumn1, DuplicateColumn2, DuplicateColumn3)

WITH TempUsers (FirstName, LastName, duplicateRecordCount)
AS
(
    SELECT FirstName, LastName,
    ROW_NUMBER() OVER (PARTITIONBY FirstName, LastName ORDERBY FirstName) AS duplicateRecordCount
    FROM dbo.Users
)
DELETE
FROM TempUsers
WHERE duplicateRecordCount > 1

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

সাধারণ টেবিল এক্সপ্রেশন জাদু।


সাবপোর্টাল / এ_হর্স_ও_নো_নাম - এটি কি কোনও আসল টেবিল থেকে নির্বাচন করা উচিত নয়? এছাড়াও, ROW_NUMBER হওয়া উচিত ROW_NUMBER () কারণ এটি একটি ফাংশন, তাই না?
ম্যাকগাইভার


1

আপনি যদি সদৃশগুলি সরানোর কোনও উপায় সন্ধান করছেন, তবুও আপনার কাছে ডুপ্লিকেটগুলি সহ টেবিলের দিকে ইশারা করার জন্য একটি বিদেশী কী রয়েছে, আপনি ধীর অথচ কার্যকর কার্সার ব্যবহার করে নীচের পদ্ধতিটি গ্রহণ করতে পারেন।

এটি বিদেশী কী টেবিলে সদৃশ কীগুলি স্থানান্তরিত করবে।

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

0
delete from person 
where ID not in
(
        select t.id from 
        (select min(ID) as id from person 
         group by email 
        ) as t
);

-1

নিচের উপায়টিও মুছে ফেলুন see

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

ফলাফল:

এখানে চিত্র বর্ণনা লিখুন

আমি জানি, এটি ছয় বছর আগে জিজ্ঞাসা করা হয়েছিল, কেবল পোস্ট করা যদি এটি কারও পক্ষে সহায়ক হয়।


-1

কোনও রান্নার সময় আপনি সংজ্ঞায়িত করতে পারেন এমন একটি পছন্দসই প্রাথমিক কী এর উপর ভিত্তি করে একটি পরিচয় কলাম রয়েছে এমন একটি সারণীতে রেকর্ডগুলি নকল করার একটি দুর্দান্ত উপায়'s আমি শুরু করার আগে আমি নীচের কোডটি ব্যবহার করে কাজ করতে একটি নমুনা ডেটা সেট পপুলেট করব:

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

এরপরে আমি একটি প্রকার তৈরি করব যার নাম কলামোনাম:

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

পরিশেষে আমি নিম্নলিখিত 3 ক্যাভ্যাট সহ একটি সঞ্চিত সংগ্রহ তৈরি করব: 1. প্রোক একটি প্রয়োজনীয় প্যারামিটার নেবে @ টেবিলনাম যা আপনার ডাটাবেসে সারণীর নামটি নির্ধারণ করবে। ২. প্রোকটির একটি বিকল্প প্যারামিটার রয়েছে ol কলামগুলি যা আপনি যে ক্ষেত্রগুলি মুছে ফেলতে চান তা পছন্দসই প্রাথমিক কী তৈরি করে ine যদি এই ক্ষেত্রটি ফাঁকা ছেড়ে যায়, তবে ধারণা করা হয় যে পরিচয় কলামের পাশাপাশি সমস্ত ক্ষেত্রগুলি পছন্দসই প্রাথমিক কী তৈরি করে। ৩. সদৃশ রেকর্ডগুলি মুছে ফেলা হলে, এটির পরিচয় কলামের সর্বনিম্ন মান সহ রেকর্ডটি বজায় রাখা হবে।

এখানে আমার মুছে ফেলা_ডিপস সংরক্ষণ করা আছে:

 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

এটি একবার মেনে চলার পরে, আপনি প্রোক চালিয়ে আপনার সমস্ত সদৃশ রেকর্ড মুছতে পারেন। কোনও পছন্দসই প্রাথমিক কীটি সংজ্ঞায়িত না করে ডুপগুলি মুছতে এই কলটি ব্যবহার করুন:

exec delete_dupes '_original'

একটি সংজ্ঞায়িত কাঙ্ক্ষিত প্রাথমিক কীটির উপর ভিত্তি করে ডুপগুলি মুছতে এই কলটি ব্যবহার করুন:

declare @table1 as columnnames
insert @table1
values ('salesrepid'),('sale')
exec delete_dupes '_original' , @table1
আমাদের সাইট ব্যবহার করে, আপনি স্বীকার করেছেন যে আপনি আমাদের কুকি নীতি এবং গোপনীয়তা নীতিটি পড়েছেন এবং বুঝতে পেরেছেন ।
Licensed under cc by-sa 3.0 with attribution required.