SQL Server에서 계단식 삭제를 어떻게 사용합니까?


332

T1과 T2라는 두 개의 테이블이 있으며 데이터가있는 기존 테이블입니다. 우리는 T1과 T2 사이에 일대 다 관계가 있습니다. T1의 레코드가 삭제 될 때 SQL Server에서 계단식 삭제를 수행하도록 테이블 정의를 변경하는 방법, T2의 모든 관련 레코드도 삭제됩니다.

외국 제약이 그들 사이에 있습니다. T2를 삭제하기 위해 테이블을 삭제하거나 트리거를 만들고 싶지 않습니다. 예를 들어 직원을 삭제하면 모든 검토 레코드도 사라져야합니다.

T1-직원,

Employee ID      
Name
Status

T2-성능 평가,

Employee ID - 2009 Review
Employee ID - 2010 Review

답변:


362

당신은,

  • 기존 외래 키 제약 조건을 삭제하고
  • ON DELETE CASCADE설정이 활성화 된 새 것을 추가하십시오 .

다음과 같은 것 :

ALTER TABLE dbo.T2
   DROP CONSTRAINT FK_T1_T2   -- or whatever it's called

ALTER TABLE dbo.T2
   ADD CONSTRAINT FK_T1_T2_Cascade
   FOREIGN KEY (EmployeeID) REFERENCES dbo.T1(EmployeeID) ON DELETE CASCADE

3
우리 팀과 나는 방금 이것을했다. 제약 조건을 삭제하고 다시 추가해야했습니다. 이것은 우리를 위해 일했습니다.
다니엘 L. VanDenBosch

2
하드 삭제를 선호하는 방법은 무엇입니까? 소프트 삭제에는 제약 조건 문제가 발생하지 않습니다. 나에게 정반대 인 것 같습니다.
Maxx

2
@Maxx 하드 삭제에서는 하나의 레코드를 삭제하고 고아 레지스터에 대해 걱정할 필요가 없지만 소프트 삭제에서는 수동으로 수행해야합니다.
호나우두 아라 우호 알베스

319

SQL Server Management Studio에서 기존 외래 키에 "Cascade delete"를 추가하려면

먼저 외래 키를 선택하고 새 쿼리 창에서 "DROP and Create To .."를 엽니 다.

여기에 이미지 설명을 입력하십시오

그런 다음 명령에 추가 ON DELETE CASCADE하십시오 ADD CONSTRAINT.

엔 이 쿼리를 실행하려면 "실행"버튼을 누르십시오.

그건 그렇고, 외래 키 목록을 얻고 "캐스케이드 삭제"가 설정된 키를 확인하려면 다음 스크립트를 실행할 수 있습니다.

SELECT 
   OBJECT_NAME(f.parent_object_id) AS 'Table name',
   COL_NAME(fc.parent_object_id,fc.parent_column_id) AS 'Field name',
   delete_referential_action_desc AS 'On Delete'
FROM sys.foreign_keys AS f,
     sys.foreign_key_columns AS fc,
     sys.tables t 
WHERE f.OBJECT_ID = fc.constraint_object_id
AND t.OBJECT_ID = fc.referenced_object_id
ORDER BY 1

그리고 당신이 할 수없는 것을 발견하면 DROP 인해 외래 키 제약 조건에 특정 테이블,하지만 당신은 FK 문제를 일으키는 작동하지 수, 당신은이 명령을 실행할 수 있습니다 :

sp_help 'TableName'

이 기사의 SQL에는 특정 테이블을 참조하는 모든 FK가 나열되어 있습니다.

이 모든 것이 도움이되기를 바랍니다.

긴 손가락에 대한 사과. 난 그냥 지적하려고했습니다.


163

SQL Server Management Studio에서이 작업을 수행 할 수 있습니다.

→ 테이블 디자인을 마우스 오른쪽 버튼으로 클릭하고 관계로 이동 한 다음 왼쪽 창과 오른쪽 창에서 외래 키를 선택하고 "INSERT and UPDATE specification"메뉴를 확장 한 다음 Delete Rule로 "Cascade"를 선택합니다.

SQL Server Management Studio


안녕하세요, 4의 차이점은 무엇입니까? 캐스케이드를 켜면 테이블의 모든 데이터를 쉽게 삭제할 수 있습니다. 이 테이블이 아니라이 테이블의 모든 dependencies / fk 키 어떻게 볼 수 있습니까 ? 모든 FK를 삭제 한 후에도 여전히 오류가 발생합니다
aggie

@aggie-의존성을 확인할 수 있습니다-테이블을 마우스 오른쪽 버튼으로 클릭-> "종속성보기"또한 SQL Server는 "DELETE 문이 REFERENCE 제약 조건"FK_Child1_Parent1 "과 충돌했습니다. 데이터베이스 "TESTDB", 테이블 "dbo.Child1", 열 'Parent1ID'에서 충돌이 발생했습니다. "
Palanikumar

@aggie-네 번째 경우 인 "기본값 설정"은 외래 키 열에 기본 제약 조건을 설정해야합니다. 부모를 삭제하면 자식 테이블에서 기본값이 바뀝니다. (참고 : 기본값은 부모 테이블과 일치해야합니다.) 자세한 내용은 mssqltips.com/sqlservertip/2365/…를
Palanikumar

이것은 매우 도움이됩니다. 삽입 규칙이없는 이유가 궁금합니다. 즉, T1에 행을 추가하면 T2의 해당 항목이 자동으로 생성되기를 원합니다.
Robert M.

@RobertM. 말이되지 않기 때문입니다. INSERT 할 값을 어떻게 알 수 있습니까? INSERT 트리거를 사용하여 자식 행을 생성하고 조사해 볼 수 있습니다.
Dan Bechard

47

같은 것을 사용하십시오

ALTER TABLE T2
ADD CONSTRAINT fk_employee
FOREIGN KEY (employeeID)
REFERENCES T1 (employeeID)
ON DELETE CASCADE;

올바른 열 이름을 입력하면 설정해야합니다. mark_s가 올바르게 언급했듯이 이미 외래 키 제약 조건이 있으면 이전 키를 먼저 삭제 한 다음 새 키를 만들어야 할 수도 있습니다.


41
@marc_s-실제로 양쪽의 동일한 열에 대해 두 번째 외래 키를 추가 할 수 있으며 올바르게 작동합니다. 가동 중지 시간이없는 프로덕션 환경에서 작업하는 경우 계단식으로 새 FK를 도입 한 다음 FK가없는 경우 테이블에 창을 남기지 않고 이전 FK를 삭제하는 것이 좋습니다. (방금 SQL 2008에서 테스트)
Damien_The_Unbeliever

맞습니다. 나는 이것을 시도하고 작동합니다. 첫 번째 외래 키 제약 조건을 삭제할 필요가 없습니다. 답장을 보내 주셔서 감사합니다.
Bichvan Nguyen

15

먼저 ONCascade 속성을 활성화하려면 :

1. 기존 외래 키 제약 조건 삭제

2. ON DELETE CASCADE 설정을 사용하여 새 것을 추가하십시오.

전의:

IF EXISTS(SELECT 1 FROM sys.foreign_keys WHERE parent_object_id = OBJECT_ID(N'dbo.Response'))
 BEGIN 

ALTER TABLE [dbo].[Response] DROP CONSTRAINT [FK_Response_Request]  

ALTER TABLE [dbo].[Response] WITH CHECK ADD CONSTRAINT [FK_Response_Request]  FOREIGN KEY([RequestId])
REFERENCES [dbo].[Request] ([RequestId])
ON DELETE CASCADE
END

ELSE

 BEGIN 
 ALTER TABLE [dbo].[Response] WITH CHECK ADD CONSTRAINT [FK_Response_Request]  FOREIGN KEY([RequestId])
REFERENCES [dbo].[Request] ([RequestId])
ON DELETE CASCADE
END

ONCascade 속성을 비활성화하는 두 번째 방법 :

1. 기존 외래 키 제약 조건 삭제

2. ON DELETE NO ACTION 설정이 활성화 된 새 항목 추가

전의:

IF EXISTS(SELECT 1 FROM sys.foreign_keys WHERE parent_object_id = OBJECT_ID(N'dbo.Response'))
 BEGIN 
ALTER TABLE [dbo].[Response] DROP CONSTRAINT [FK_Response_Request]  

ALTER TABLE [dbo].[Response] WITH CHECK ADD CONSTRAINT [FK_Response_Request]  FOREIGN KEY([RequestId])
REFERENCES [dbo].[Request] ([RequestId])
ON DELETE CASCADE
END

ELSE

 BEGIN 
 ALTER TABLE [dbo].[Response] WITH CHECK ADD CONSTRAINT [FK_Response_Request]  FOREIGN KEY([RequestId])
REFERENCES [dbo].[Request] ([RequestId])
ON DELETE NO ACTION 
END

15

ON DELETE CASCADE
상위 데이터가 삭제 될 때 하위 데이터가 삭제되도록 지정합니다.

CREATE TABLE products
( product_id INT PRIMARY KEY,
  product_name VARCHAR(50) NOT NULL,
  category VARCHAR(25)
);

CREATE TABLE inventory
( inventory_id INT PRIMARY KEY,
  product_id INT NOT NULL,
  quantity INT,
  min_level INT,
  max_level INT,
  CONSTRAINT fk_inv_product_id
    FOREIGN KEY (product_id)
    REFERENCES products (product_id)
    ON DELETE CASCADE
);

이 외래 키 ON DELETE CASCADE에 대해 부모 테이블의 데이터가 삭제 될 때 자식 테이블의 해당 레코드를 삭제하도록 SQL Server에 지시 하는 절을 지정했습니다 . 따라서이 예제에서 product_id 값이 products 테이블에서 삭제되면이 product_id를 사용하는 인벤토리 테이블의 해당 레코드도 삭제됩니다.


-2

일대 다 관계가 T1에서 T2까지 인 경우 함수를 나타내지 않으므로 결과 T2 값이 연역적으로 유효한 T1 결합 T2의 튜플을 생략하지 않도록 보장하는 역함수를 추론하거나 추론하는 데 사용할 수 없습니다 연역적으로 유효한 역함수가 없기 때문입니다. (기능을 나타내는 것이 기본 키의 목적이었습니다.) SQL의 대답은 예라고 할 수 있습니다. 관계형 사고의 대답은 할 수 없다는 것입니다. Codd 1970의 모호한 점을 참조하십시오. 관계는 T1에서 T2까지 다 대일이어야합니다.


-10

실제 속성 데이터 인 경우 테이블 속성을 삭제할 수는 없으며 테이블 스키마에 영향을 미치지 않는 내용을 삭제하십시오.

당사 사이트를 사용함과 동시에 당사의 쿠키 정책개인정보 보호정책을 읽고 이해하였음을 인정하는 것으로 간주합니다.
Licensed under cc by-sa 3.0 with attribution required.