다음을 통해 공유


테이블 힌트(Transact-SQL)

테이블 힌트에서 잠금 메서드, 하나 이상의 인덱스, Table Scan 또는 Index Seek와 같은 쿼리 처리 연산이나 기타 옵션을 지정하여 DML(데이터 조작 언어) 문이 실행되는 동안 쿼리 최적화 프로그램의 기본 동작을 재정의합니다. 테이블 힌트는 DML 문의 FROM 절에 지정되어 해당 절에서 참조하는 테이블 또는 뷰에만 영향을 미칩니다.

주의 사항주의

일반적으로 SQL Server 쿼리 최적화 프로그램은 쿼리에 대해 최상의 실행 계획을 선택하므로 숙련된 개발자 및 데이터베이스 관리자가 마지막 방법으로만 힌트를 사용하는 것이 좋습니다.

적용 대상

DELETE

INSERT

SELECT

UPDATE

MERGE

항목 링크 아이콘Transact-SQL 구문 표기 규칙

구문

WITH  ( <table_hint> [ [, ]...n ] )

<table_hint> ::= 
[ NOEXPAND ] { 
    INDEX ( index_value [ ,...n ] ) | INDEX =  ( index_value )
  | FASTFIRSTROW 
  | FORCESEEK [( index_value ( index_column_name  [ ,... ] ) ) ]
  | FORCESCAN
  | HOLDLOCK 
  | NOLOCK 
  | NOWAIT
  | PAGLOCK 
  | READCOMMITTED 
  | READCOMMITTEDLOCK 
  | READPAST 
  | READUNCOMMITTED 
  | REPEATABLEREAD 
  | ROWLOCK 
  | SERIALIZABLE 
  | TABLOCK 
  | TABLOCKX 
  | UPDLOCK 
  | XLOCK 
} 

<table_hint_limited> ::=
{
    KEEPIDENTITY 
  | KEEPDEFAULTS 
  | FASTFIRSTROW 
  | HOLDLOCK 
  | IGNORE_CONSTRAINTS 
  | IGNORE_TRIGGERS 
  | NOLOCK 
  | NOWAIT
  | PAGLOCK 
  | READCOMMITTED 
  | READCOMMITTEDLOCK 
  | READPAST 
  | REPEATABLEREAD 
  | ROWLOCK 
  | SERIALIZABLE 
  | TABLOCK 
  | TABLOCKX 
  | UPDLOCK 
  | XLOCK 
} 

인수

  • WITH ( <table_hint> ) [ [ , ]...n ]
    몇 가지 예외가 있지만 테이블 힌트는 WITH 키워드를 사용하여 힌트를 지정할 때만 지원됩니다. 괄호가 필요합니다.

    중요 정보중요

    WITH 키워드 생략은 사용되지 않는 기능이며 다음 버전의 Microsoft SQL Server에서 제거됩니다. 항상 새 개발 작업에서 WITH를 지정하고 이 키워드가 현재 생략된 응용 프로그램을 수정하십시오.

    쉼표가 아닌 공백으로 구분하는 힌트는 사용되지 않는 기능이며 다음 버전의 Microsoft SQL Server에서 제거됩니다. 항상 새 개발 작업에서 쉼표를 지정하고 이 쉼표가 현재 생략된 응용 프로그램을 수정하십시오.

    다음 테이블 힌트는 WITH 키워드와 함께 또는 WITH 키워드 없이 사용 가능합니다. READUNCOMMITTED, UPDLOCK, REPEATABLEREAD, SERIALIZABLE, READCOMMITTED, FASTFIRSTROW, TABLOCK, TABLOCKX, PAGLOCK, ROWLOCK, NOWAIT, READPAST, XLOCK 및 NOEXPAND 테이블 힌트를 WITH 키워드와 함께 또는 WITH 키워드 없이 사용할 수 있습니다. 이러한 테이블 힌트를 WITH 키워드 없이 지정하는 경우 힌트를 단독으로 지정해야 합니다. 예: FROM t WITH (TABLOCK). 힌트를 다른 옵션과 함께 지정하는 경우 힌트를 WITH 키워드로 지정해야 합니다. 예: FROM t WITH (TABLOCK, INDEX(myindex)).

    호환성 수준이 90 이상인 데이터베이스에 대한 쿼리에 힌트를 사용할 경우 제한 사항이 적용됩니다.

  • NOEXPAND
    쿼리 최적화 프로그램에서 쿼리를 처리할 때 기본 테이블에 액세스하기 위해 인덱싱된 뷰를 확장하지 않도록 지정합니다. 쿼리 최적화 프로그램은 뷰를 클러스터형 인덱스가 있는 테이블처럼 처리합니다. NOEXPAND는 인덱싱된 뷰에만 적용됩니다. 자세한 내용은 설명 부분을 참조하십시오.

  • INDEX (index_value [,... n ] ) | INDEX = (index_value)
    INDEX(index_value) 구문은 쿼리 최적화 프로그램이 문을 처리할 때 사용할 하나 이상의 인덱스 이름이나 ID를 지정합니다. 대체 구문 INDEX = (index_value)는 단일 인덱스 값만 허용합니다.

    클러스터형 인덱스가 있는 경우 INDEX(0)에서 클러스터형 인덱스 검색을 강제 실행하고 INDEX(1)에서 클러스터형 인덱스 검색 또는 찾기를 강제 실행합니다. 클러스터형 인덱스가 없는 경우 INDEX(0)은 테이블 검색을 강제 실행하고 INDEX(1)은 오류로 해석됩니다.

    단일 힌트 목록에서 여러 인덱스가 사용되는 경우 모든 중복 사항은 무시되고 나열된 인덱스 중 나머지를 사용하여 테이블의 행을 검색합니다. 이때 인덱스 힌트에 있는 인덱스의 순서가 중요합니다. 여러 인덱스 힌트는 또한 인덱스 AND 연산을 강제 실행하고 쿼리 최적화 프로그램은 액세스되는 각 인덱스에 대해 가능한 한 많은 조건을 적용합니다. 인덱스 힌트 컬렉션에 쿼리에서 참조하는 모든 열이 포함되지 않은 경우 SQL Server 데이터베이스 엔진에서 인덱싱된 열을 모두 검색한 후 인출을 수행하여 남은 열을 검색합니다.

    [!참고]

    여러 인덱스를 참조하는 인덱스 힌트가 스타 조인의 팩트 테이블에 사용되는 경우 최적화 프로그램은 인덱스 힌트를 무시하고 경고 메시지를 반환합니다. 또한 인덱스 힌트가 지정된 테이블에 대해서는 인덱스 OR 연산을 사용할 수 없습니다.

    테이블 힌트에서 지정할 수 있는 최대 인덱스 수는 250개의 비클러스터형 인덱스입니다.

  • KEEPIDENTITY
    OPENROWSET에 BULK 옵션을 사용하는 경우 INSERT 문에만 적용됩니다.

    가져온 데이터 파일의 ID 값이 ID 열에 사용되도록 지정합니다. KEEPIDENTITY를 지정하지 않는 경우 이 열의 ID 값을 확인하지만 가져오지는 않으며 쿼리 최적화 프로그램은 테이블 생성 중에 지정된 초기값 및 증가값에 따라 고유 값을 자동으로 할당합니다.

    중요 정보중요

    데이터 파일의 테이블 또는 뷰에서 ID 열에 값이 없고 ID 열이 테이블의 마지막 열이 아닌 경우 ID 열을 건너뛰어야 합니다. 자세한 내용은 서식 파일을 사용하여 데이터 필드 건너뛰기를 참조하십시오. ID 열을 성공적으로 건너뛰면 쿼리 최적화 프로그램은 가져온 테이블 행에 ID 열의 고유 값을 자동으로 할당합니다.

    INSERT ... SELECT * FROM OPENROWSET(BULK...) 문에서 이 힌트를 사용하는 예는 데이터 대량 가져오기 중 ID 값 유지를 참조하십시오.

    테이블의 ID 값을 확인하는 방법은 DBCC CHECKIDENT(Transact-SQL)를 참조하십시오.

  • KEEPDEFAULTS
    OPENROWSET에 BULK 옵션을 사용하는 경우 INSERT 문에만 적용됩니다.

    데이터 레코드의 열에 값이 없는 경우 NULL 대신 테이블 열의 기본값(있는 경우)을 삽입하도록 지정합니다.

    INSERT ... SELECT <columns> FROM OPENROWSET(BULK...) 문에서 이 힌트를 사용하는 예는 대량 가져오기 수행 중 Null 유지 또는 기본값 사용를 참조하십시오.

  • FASTFIRSTROW
    OPTION(FAST 1)과 동일합니다. 자세한 내용은 쿼리 힌트(Transact-SQL)를 참조하십시오.

    중요 정보중요

    이 기능은 다음 버전의 Microsoft SQL Server에서 제거됩니다. 새 개발 작업에서는 이 기능을 사용하지 말고, 현재 이 기능을 사용하는 응용 프로그램은 가능한 한 빨리 수정하십시오.

  • FORCESEEK [ **(index_value(**index_column_name [ ,... n ] )) ]
    쿼리 최적화 프로그램이 테이블 또는 뷰의 데이터에 대한 액세스 경로로 Index Seek 연산만 사용하도록 지정합니다. SQL Server 2008 R2 SP1부터 인덱스 매개 변수를 지정할 수도 있습니다. 인덱스 매개 변수와 함께 FORCESEEK를 지정하는 것은 INDEX 힌트와 함께 FORCESEEK를 사용하는 것과 유사합니다. 그러나 검색할 인덱스와 Seek 연산에서 고려하는 인덱스 열을 모두 지정하여 쿼리 최적화 프로그램에서 사용된 액세스 경로를 더 많이 제어할 수 있습니다.

    • index_value
      인덱스 이름 또는 인덱스 ID 값입니다. 인덱스 ID 0(힙)을 지정할 수 없습니다. 인덱스 이름 또는 ID를 반환하려면 sys.indexes 카탈로그 뷰를 쿼리하십시오.

    • index_column_name
      Seek 연산에 포함하려는 인덱스 열의 이름입니다. 쿼리 최적화 프로그램에서는 지정된 인덱스 열을 최소로 사용하는 지정된 인덱스를 통해 Index Seek 연산만 고려합니다. 최적화 프로그램은 필요할 경우 추가 열을 고려할 수도 있습니다. 예를 들어 비클러스터형 인덱스를 지정하는 경우 최적화 프로그램에서 지정된 열과 함께 클러스터형 인덱스 키 열을 사용하도록 선택할 수도 있습니다.

    FORCESEEK 힌트는 다음과 같은 방법으로 지정할 수 있습니다.

    구문

    설명

    인덱스 또는 INDEX 힌트 없이 지정

    FROM dbo.MyTable WITH (FORCESEEK)

    쿼리 최적화 프로그램에서는 관련된 모든 인덱스를 통해 테이블 또는 뷰에 액세스하기 위해 Index Seek 연산만 고려합니다.

    INDEX 힌트와 함께 지정

    FROM dbo.MyTable WITH (FORCESEEK, INDEX (MyIndex))

    쿼리 최적화 프로그램에서는 지정된 인덱스를 통해 테이블 또는 뷰에 액세스하기 위해 Index Seek 연산만 고려합니다.

    인덱스 및 인덱스 열을 지정하여 매개 변수화 지정

    FROM dbo.MyTable WITH (FORCESEEK (MyIndex (col1, col2, col3)))

    쿼리 최적화 프로그램에서는 지정된 인덱스 열을 최소로 사용하는 지정된 인덱스를 통해 테이블 또는 뷰에 액세스하기 위해 Index Seek 연산만 고려합니다.

    FORCESEEK 힌트(인덱스 매개 변수와 함께 또는 없이 사용)를 사용할 경우 다음 지침을 고려합니다.

    • 힌트는 테이블 힌트 또는 쿼리 힌트로 지정될 수 있습니다. 쿼리 힌트에 대한 자세한 내용은 쿼리 힌트(Transact-SQL)를 참조하십시오.

    • FORCESEEK를 인덱싱된 뷰로 적용하려면 NOEXPAND 힌트도 지정해야 합니다.

    • 힌트는 테이블 또는 뷰마다 최대 한 번 적용될 수 있습니다.

    • 힌트는 원격 데이터 원본에 대해 지정할 수 없습니다. FORCESEEK가 인덱스 힌트와 함께 지정되면 7377 오류가 반환되고 FORCESEEK가 인덱스 힌트 없이 사용되면 8180 오류가 반환됩니다.

    • FORCESEEK로 계획이 검색되지 않으면 8622 오류가 반환됩니다.

    FORCESEEK가 인덱스 매개 변수와 함께 지정된 경우 다음 지침과 제한 사항이 적용됩니다.

    • 힌트는 INDEX 힌트 또는 기타 FORCESEEK 힌트 중 하나와 함께 지정될 수 없습니다.

    • 최소 하나의 열이 지정되어야 하고 선행 키 열이어야 합니다.

    • 추가 인덱스 열을 지정할 수 있으나 키 열은 건너뛸 수 없습니다. 예를 들어, 지정된 인덱스에 키 열 a, b, c가 포함되는 경우 유효한 구문에 FORCESEEK (MyIndex (a)) 및 FORCESEEK (MyIndex (a, b)가 포함됩니다. 유효하지 않은 구문에는 FORCESEEK (MyIndex (c)) 및 FORCESEEK (MyIndex (a, c)가 포함됩니다.

    • 힌트에 지정된 열 이름 순서는 참조된 인덱스의 열 순서와 일치해야 합니다.

    • 인덱스 키 정의에 없는 열은 지정할 수 없습니다. 예를 들어 비클러스터형 인덱스에서 정의된 인덱스 키 열만 지정할 수 있습니다. 인덱스에 자동으로 포함된 클러스터형 키 열은 지정될 수 없으나 최적화 프로그램에서 사용될 수도 있습니다.

    • 인덱스 정의를 수정하면(예: 열 추가 또는 제거) 해당 인덱스를 참조하는 쿼리를 수정해야 할 수도 있습니다.

    • 힌트를 사용하면 최적화 프로그램에서 테이블의 모든 공간 또는 XML 인덱스를 고려하지 않아도 됩니다.

    • 힌트는 FORCESCAN 힌트와 함께 지정할 수 없습니다.

    • 분할된 인덱스의 경우 SQL Server에서 암시적으로 추가된 분할 열을 FORCESEEK 힌트에 지정할 수 없습니다. 자세한 내용은 분할 인덱스에 대한 특수 지침을 참조하십시오.

    주의 사항주의

    인덱스 매개 변수와 함께 FORCESEEK를 지정하면 매개 변수 없이 FORCESEEK를 지정할 때보다 최적화 프로그램에서 고려할 수 있는 계획 수가 더 제한됩니다. 이로 인해 "계획을 생성할 수 없음" 오류가 더 많이 발생할 수 있습니다. 후속 릴리스에서 최적화 프로그램을 내부 수정하면 보다 많은 계획을 고려할 수도 있습니다. 자세한 내용은 FORCESEEK 테이블 힌트 사용을 참조하십시오.

  • FORCESCAN
    SQL Server 2008 R2 SP1에서 소개된 이번 힌트에서는 쿼리 최적화 프로그램이 인덱스 검색 작업만 참조된 테이블 또는 뷰에 대한 액세스 경로로 사용하도록 지정합니다. FORCESCAN 힌트는 최적화 프로그램이 영향을 받는 행 수를 과소 평가하고 검색 작업 대신 Seek 연산자를 선택하는 쿼리에 유용할 수 있습니다. 이러한 경우 작업에 부여된 메모리 양이 너무 작고 쿼리 성능이 영향을 받습니다.

    FORCESCAN은 인덱스 힌트와 함께 또는 인덱스 힌트 없이 지정할 수 있습니다. 인덱스 힌트(INDEX = index_name, FORCESCAN)와 함께 사용할 경우 쿼리 최적화 프로그램에서는 참조된 테이블에 액세스할 때 지정된 인덱스를 통한 액세스 경로 검색만 고려합니다. 기본 테이블에 대한 테이블 검색 작업을 강제 실행하기 위해 FORCESCAN을 인덱스 힌트 INDEX(0)와 함께 지정할 수 없습니다.

    분할된 테이블과 인덱스의 경우 FORCESCAN은 쿼리 조건자 평가를 통해 파티션이 제거된 후에 적용됩니다. 즉, 검색은 나머지 파티션에만 적용되고 전체 테이블에는 적용되지 않습니다.

    FORCESCAN 힌트는 다음과 같은 제한 사항이 있습니다.

    • INSERT, UPDATE 또는 DELETE 문의 대상인 테이블에는 힌트를 지정할 수 없습니다.

    • 힌트를 두 개 이상의 인덱스 힌트와 함께 사용할 수 없습니다.

    • 힌트를 사용하면 최적화 프로그램에서 테이블의 모든 공간 또는 XML 인덱스를 고려하지 않아도 됩니다.

    • 힌트는 원격 데이터 원본에 대해 지정할 수 없습니다.

    • 힌트는 FORCESEEK 힌트와 함께 지정할 수 없습니다.

  • HOLDLOCK
    SERIALIZABLE과 동일합니다. 자세한 내용은 이 항목의 뒷부분에 나오는 SERIALIZABLE을 참조하십시오. HOLDLOCK은 이 옵션이 지정된 테이블 또는 뷰에 대해 이 옵션이 사용된 문이 정의한 트랜잭션이 완료될 때까지만 적용됩니다. HOLDLOCK은 FOR BROWSE 옵션이 포함된 SELECT 문에 사용할 수 없습니다.

  • IGNORE_CONSTRAINTS
    OPENROWSET에 BULK 옵션을 사용하는 경우 INSERT 문에만 적용됩니다.

    대량 가져오기 작업에서 테이블의 모든 제약 조건을 무시하도록 지정합니다. 기본적으로 INSERT는 CHECKFOREIGN KEY 제약 조건을 검사하지만 대량 가져오기 작업에 IGNORE_CONSTRAINTS가 지정된 경우 INSERT는 대상 테이블의 이러한 제약 조건을 무시해야 합니다. UNIQUE, PRIMARY KEY 또는 NOT NULL 제약 조건은 비활성화할 수 없습니다.

    입력 데이터에 제약 조건을 위반하는 행이 포함된 경우에는 CHECK 및 FOREIGN KEY 제약 조건을 비활성화할 수 있습니다. CHECK 및 FOREIGN KEY 제약 조건을 비활성화하여 데이터를 가져온 다음 Transact-SQL 문을 사용하여 데이터를 정리할 수 있습니다.

    하지만 CHECK 및 FOREIGN KEY 제약 조건이 무시된 경우 작업 이후 무시된 각 테이블 제약 조건은 sys.check_constraints 또는 sys.foreign_keys 카탈로그 뷰에 is_not_trusted로 표시됩니다. 어느 시점에서는 전체 테이블의 제약 조건을 확인해야 합니다. 대량 가져오기 작업을 수행하기 전에 테이블이 비어 있지 않으면 제약 조건의 유효성을 다시 검사하는 비용이 증분 데이터에 CHECK 및 FOREIGN KEY 제약 조건을 적용하는 비용을 초과할 수 있습니다.

  • IGNORE_TRIGGERS
    OPENROWSET에 BULK 옵션을 사용하는 경우 INSERT 문에만 적용됩니다.

    대량 가져오기 작업에서 테이블에 정의된 모든 트리거를 무시하도록 지정합니다. 기본적으로 INSERT는 트리거를 적용합니다.

    응용 프로그램이 트리거에 의존하지 않으며 성능을 최대화해야 하는 경우에만 IGNORE_TRIGGERS를 사용하십시오.

  • NOLOCK
    READUNCOMMITTED와 동일합니다. 자세한 내용은 이 항목의 뒷부분에 나오는 READUNCOMMITTED를 참조하십시오.

    [!참고]

    UPDATE 또는 DELETE 문의 경우: Microsoft SQL Server의 이후 버전에서는 이 기능이 제거됩니다. 새 개발 작업에서는 이 기능을 사용하지 않도록 하고, 현재 이 기능을 사용하는 응용 프로그램은 수정하십시오.

  • NOWAIT
    테이블에 잠금이 있으면 데이터베이스 엔진에서 바로 메시지를 반환하도록 지정합니다. NOWAIT는 특정 테이블에 SET LOCK_TIMEOUT 0을 지정하는 것과 동일합니다.

  • PAGLOCK
    일반적으로 행 또는 키에 개별 잠금이 사용되거나 일반적으로 단일 테이블 잠금이 사용되는 곳에서 페이지 잠금을 사용합니다. 기본적으로 작업에 적합한 잠금 모드를 사용합니다. SNAPSHOT 격리 수준에서 작동하는 트랜잭션에 지정하는 경우 UPDLOCK 및 HOLDLOCK과 같은 잠금이 필요한 다른 테이블 힌트와 함께 PAGLOCK을 사용하지 않으면 페이지 잠금이 수행되지 않습니다.

  • READCOMMITTED
    읽기 작업이 잠금 또는 행 버전 관리를 사용하여 READ COMMITTED 격리 수준에 대한 규칙을 따르도록 지정합니다. READ_COMMITTED_SNAPSHOT 데이터베이스 옵션이 OFF인 경우 데이터베이스 엔진은 데이터를 읽을 때 공유 잠금을 획득하고 읽기 작업이 완료되면 잠금을 해제합니다. READ_COMMITTED_SNAPSHOT 데이터베이스 옵션이 ON인 경우 데이터베이스 엔진은 잠금을 획득하지 않고 행 버전 관리를 사용합니다. 격리 수준에 대한 자세한 내용은 SET TRANSACTION ISOLATION LEVEL(Transact-SQL)을 참조하십시오.

    [!참고]

    UPDATE 또는 DELETE 문의 경우: Microsoft SQL Server의 이후 버전에서는 이 기능이 제거됩니다. 새 개발 작업에서는 이 기능을 사용하지 않도록 하고, 현재 이 기능을 사용하는 응용 프로그램은 수정하십시오.

  • READCOMMITTEDLOCK
    읽기 작업이 잠금을 사용하여 READ COMMITTED 격리 수준에 대한 규칙을 따르도록 지정합니다. 데이터베이스 엔진은 READ_COMMITTED_SNAPSHOT 데이터베이스 옵션 설정에 관계없이 데이터를 읽을 때 공유 잠금을 획득하고 읽기 작업이 완료되면 잠금을 해제합니다. 격리 수준에 대한 자세한 내용은 SET TRANSACTION ISOLATION LEVEL(Transact-SQL)을 참조하십시오. INSERT 문의 대상 테이블에 힌트를 지정할 수 없습니다. 4140 오류가 반환됩니다.

  • READPAST
    데이터베이스 엔진이 다른 트랜잭션에 의해 잠긴 행을 읽지 않도록 지정합니다. READPAST가 지정되면 행 수준 잠금을 건너뜁니다. 즉, 데이터베이스 엔진은 잠금이 해제될 때까지 현재 트랜잭션을 차단하는 대신 행을 건너뜁니다. 예를 들어 T1 테이블에 값이 1, 2, 3, 4, 5인 단일 정수 열이 있다고 가정합니다. 트랜잭션 A가 값 3을 8로 변경했으나 아직 커밋하지 않은 경우 SELECT * FROM T1 (READPAST)는 1, 2, 4, 5 값을 생성합니다. READPAST는 주로 SQL Server 테이블을 사용하는 작업 큐를 구현할 때 잠금 경합을 줄이는 데 사용됩니다. READPAST를 사용하는 큐 판독기는 다른 트랜잭션에 의해 잠긴 큐 항목이 있으면 다른 트랜잭션이 잠금을 해제할 때까지 기다리지 않고 사용 가능한 다음 큐 항목으로 건너뜁니다.

    READPAST는 UPDATE 또는 DELETE 문에서 참조되는 테이블과 FROM 절에서 참조되는 테이블에 지정할 수 있습니다. UPDATE 문에 지정되는 경우 READPAST는 문에서 지정된 위치와 관계없이 데이터를 읽어 업데이트할 레코드를 식별하는 경우에만 적용됩니다. READPAST는 INSERT 문의 INTO 절에서 테이블에 대해 지정할 수 없습니다. READPAST를 사용하는 읽기 작업은 차단되지 않습니다. READPAST를 사용하는 업데이트 또는 삭제 작업은 외래 키 또는 인덱싱된 뷰를 읽거나 보조 인덱스를 수정할 때 차단될 수 있습니다.

    READPAST는 READ COMMITTED 또는 REPEATABLE READ 격리 수준에서 작동하는 트랜잭션에만 지정할 수 있습니다. SNAPSHOT 격리 수준에서 작동하는 트랜잭션에 지정하는 경우 UPDLOCK 및 HOLDLOCK과 같은 잠금이 필요한 다른 테이블 힌트와 함께 READPAST를 사용해야 합니다.

    READ_COMMITTED_SNAPSHOT 데이터베이스 옵션이 ON으로 설정되어 있고 다음 조건 중 하나에 해당하는 경우에는 READPAST 테이블 힌트를 지정할 수 없습니다.

    • 세션의 트랜잭션 격리 수준이 READ COMMITTED입니다.

    • READCOMMITTED 테이블 힌트가 쿼리에도 지정되어 있습니다.

    이러한 경우 READPAST 힌트를 지정하려면 READCOMMITTED 테이블 힌트가 있으면 이를 제거하고 쿼리에 READCOMMITTEDLOCK 테이블 힌트를 포함합니다.

  • READUNCOMMITTED
    더티 읽기를 허용하도록 지정합니다. 다른 트랜잭션이 현재 트랜잭션의 데이터 읽기를 수정하지 못하도록 하는 공유 잠금이 실행되지 않으며 다른 트랜잭션에 의해 설정된 배타적 잠금은 현재 트랜잭션의 잠긴 데이터 읽기를 차단하지 않습니다. 더티 읽기를 허용하면 동시성이 높아질 수 있지만 다른 트랜잭션이 데이터 수정 내용을 읽고 롤백할 가능성이 있습니다. 이 경우 트랜잭션에 대한 오류가 생성되거나 사용자에게 커밋되지 않은 데이터가 제공될 수 있습니다. 또한 사용자에게 레코드가 두 번 표시되거나 전혀 표시되지 않을 수도 있습니다. 더티 읽기, 반복할 수 없는 읽기 및 가상 읽기에 대한 자세한 내용은 동시성 효과를 참조하십시오.

    READUNCOMMITTED 및 NOLOCK 힌트는 데이터 잠금에만 적용됩니다. READUNCOMMITTED 및 NOLOCK 힌트가 있는 쿼리를 포함하여 모든 쿼리는 컴파일 및 실행 중에 Sch-S(스키마 안정성) 잠금을 획득합니다. 이 때문에 동시 트랜잭션이 테이블에 대해 Sch-M(스키마 수정) 잠금을 유지하면 쿼리가 차단됩니다. 예를 들어 DDL(데이터 정의 언어) 작업은 테이블의 스키마 정보를 수정하기 전에 Sch-M 잠금을 획득합니다. READUNCOMMITTED 또는 NOLOCK 힌트를 사용하여 실행되는 쿼리를 포함하여 Sch-S 잠금을 획득하려고 시도하는 동시 쿼리는 모두 차단됩니다. 반대로 Sch-S 잠금을 유지하는 쿼리는 Sch-M 잠금을 획득하려고 시도하는 동시 트랜잭션을 차단합니다. 잠금 동작에 대한 자세한 내용은 잠금 호환성(데이터베이스 엔진)을 참조하십시오.

    삽입, 업데이트 또는 삭제 작업으로 수정된 테이블에 대해서는 READUNCOMMITTED와 NOLOCK을 지정할 수 없습니다. SQL Server 쿼리 최적화 프로그램은 UPDATE 또는 DELETE 문의 대상 테이블에 적용되는 FROM 절의 READUNCOMMITTED 및 NOLOCK 힌트를 무시합니다.

    [!참고]

    UPDATE 또는 DELETE 문의 대상 테이블에 적용되는 FROM 절의 READUNCOMMITTED 및 NOLOCK 힌트 사용 지원은 SQL Server의 이후 버전에서 제거될 예정입니다. 새 개발 작업에서는 이 컨텍스트에서 이러한 힌트를 사용하지 않도록 하고 현재 이 힌트를 사용하는 응용 프로그램은 수정하십시오.

    다음 중 하나를 사용하여 트랜잭션에서 커밋되지 않은 데이터 수정 내용에 대해 더티 읽기를 수행할 수 없도록 하여 잠금 경합을 최소화할 수도 있습니다.

    • READ_COMMITTED_SNAPSHOT 데이터베이스 옵션이 ON으로 설정된 READ COMMITTED 격리 수준

    • SNAPSHOT 격리 수준

    격리 수준에 대한 자세한 내용은 SET TRANSACTION ISOLATION LEVEL(Transact-SQL)을 참조하십시오.

    [!참고]

    READUNCOMMITTED를 지정할 때 오류 메시지 601이 표시되면 교착 상태 오류(1205)를 해결하는 방법으로 오류를 해결하고 문을 다시 시도하십시오.

  • REPEATABLEREAD
    REPEATABLE READ 격리 수준에서 실행되는 트랜잭션과 동일한 잠금 기능으로 검색이 수행되도록 지정합니다. 격리 수준에 대한 자세한 내용은 SET TRANSACTION ISOLATION LEVEL(Transact-SQL)을 참조하십시오.

  • ROWLOCK
    페이지 또는 테이블 잠금이 일반적으로 사용될 때 행 잠금을 사용하도록 지정합니다. SNAPSHOT 격리 수준에서 작동하는 트랜잭션에 지정하는 경우 UPDLOCK 및 HOLDLOCK과 같은 잠금이 필요한 다른 테이블 힌트와 함께 ROWLOCK을 사용하지 않으면 행 잠금이 수행되지 않습니다.

  • SERIALIZABLE
    HOLDLOCK과 동일합니다. 필요한 테이블 또는 데이터 페이지가 더 이상 필요 없을 때 트랜잭션의 완료 여부와 관계없이 즉시 공유 잠금을 해제하지 않고 트랜잭션 완료 시까지 유지함으로써 공유 잠금을 더욱 제한적으로 만듭니다. SERIALIZABLE 격리 수준에서 실행되는 트랜잭션과 동일한 잠금 기능으로 검색이 수행됩니다. 격리 수준에 대한 자세한 내용은 SET TRANSACTION ISOLATION LEVEL(Transact-SQL)을 참조하십시오.

  • TABLOCK
    획득한 잠금이 테이블 수준에 적용되도록 지정합니다. 획득한 잠금 유형은 실행 중인 문에 따라 달라집니다. 예를 들어, SELECT 문에서는 공유 잠금을 획득할 수도 있습니다. TABLOCK을 지정하여 공유 잠금을 행 또는 페이지 수준 대신 전체 테이블에 적용합니다. HOLDLOCK도 함께 지정한 경우 테이블 잠금이 트랜잭션이 끝날 때까지 유지됩니다.

    INSERT INTO <target_table> SELECT <columns> FROM <source_table> 문을 사용하여 힙으로 데이터를 가져올 때 대상 테이블에 대해 TABLOCK 힌트를 지정하여 문에 대해 잠금과 최적화된 로깅을 활성화할 수 있습니다. 또한 데이터베이스의 복구 모델은 단순 또는 대량으로 지정해야 합니다. 자세한 내용은 INSERT(Transact-SQL)를 참조하십시오.

    OPENROWSET 대량 행 집합 공급자와 함께 TABLOCK을 사용하여 테이블로 데이터를 가져오는 경우 여러 클라이언트가 로깅 및 잠금이 최적화된 상태에서 동시에 대상 테이블로 데이터를 로드할 수 있습니다. 자세한 내용은 대량 가져오기의 최소 로깅을 위한 선행 조건을 참조하십시오.

  • TABLOCKX
    테이블을 배타적으로 잠그도록 지정합니다.

  • UPDLOCK
    업데이트 잠금을 사용하고 트랜잭션이 완료될 때까지 유지하도록 지정합니다. UPDLOCK은 읽기 작업을 위해 행 수준 또는 페이지 수준에서만 업데이트 잠금을 사용합니다. UPDLOCK이 TABLOCK과 함께 사용되는 경우 또는 테이블 수준 잠금이 여러 이유로 인해 사용되는 경우, 배타(X) 잠금이 대신 사용됩니다.

    UPDLOCK이 지정된 경우 READCOMMITTED 및 READCOMMITTEDLOCK 격리 수준 힌트는 무시됩니다. 예를 들어, 세션의 격리 수준이 SERIALIZABLE로 설정되고 쿼리가 (UPDLOCK, READCOMMITTED)를 지정할 경우 READCOMMITTED 힌트는 무시되고 트랜잭션은 SERIALIZABLE 격리 수준을 사용하여 실행됩니다.

  • XLOCK
    배타적 잠금을 사용하고 트랜잭션이 완료될 때까지 유지하도록 지정합니다. ROWLOCK, PAGLOCK 또는 TABLOCK과 함께 지정한 경우에는 배타적 잠금이 적절한 세분성 수준에 적용됩니다.

주의

테이블을 쿼리 계획으로 액세스하지 않는 경우에는 테이블 힌트가 무시됩니다. 최적화 프로그램의 선택에 의해 테이블 자체를 액세스하지 않거나 테이블 대신 인덱싱된 뷰를 액세스하기로 하는 경우 등이 해당됩니다. 후자의 경우에 OPTION(EXPAND VIEWS) 쿼리 힌트를 사용하여 인덱싱된 뷰 액세스를 금지할 수도 있습니다.

모든 잠금 힌트는 뷰에서 참조되는 테이블과 뷰를 포함하여 쿼리 계획에서 액세스하는 모든 테이블과 뷰로 전달됩니다. 또한 SQL Server는 해당되는 잠금 일관성 검사를 수행합니다.

행 수준의 잠금을 획득하는 잠금 힌트 ROWLOCK, UPDLOCK 및 XLOCK은 실제 데이터 행이 아닌 인덱스 키에 잠금을 설정합니다. 예를 들어 테이블에 비클러스터형 인덱스가 있고 잠금 힌트를 사용하는 SELECT 문이 포함 인덱스에 의해 처리되는 경우 잠금은 기본 테이블의 데이터 행이 아닌 포함 인덱스의 인덱스 키에서 획득됩니다.

테이블에 계산 열이 있고 계산 열이 다른 테이블의 열에 액세스하는 식 또는 함수에 의해 계산되는 경우 테이블 힌트는 해당 테이블에서 사용되지 않습니다. 즉, 테이블 힌트가 전파되지 않습니다. 예를 들어 NOLOCK 테이블 힌트는 쿼리의 테이블에 지정됩니다. 이 테이블에는 다른 테이블에 있는 열에 액세스하는 식과 함수의 조합에 의해 계산되는 계산 열이 있습니다. 이러한 식과 함수에 의해 참조되는 테이블은 액세스될 때 NOLOCK 테이블 힌트를 사용하지 않습니다.

SQL Server는 FROM 절의 각 테이블에 대해 다음 각 그룹에서 두 개 이상의 테이블 힌트를 허용하지 않습니다.

  • 세분성 힌트: PAGLOCK, NOLOCK, READCOMMITTEDLOCK, ROWLOCK, TABLOCK, 또는 TABLOCKX.

  • 격리 수준 힌트: HOLDLOCK, NOLOCK, READCOMMITTED, REPEATABLEREAD, SERIALIZABLE

필터링된 인덱스 힌트

필터링된 인덱스는 테이블 힌트로 사용될 수 있지만 쿼리가 선택한 모든 행을 포함하지 않을 경우 쿼리 최적화 프로그램에서 오류 8622가 발생합니다. 다음은 잘못된 필터링된 인덱스 힌트에 대한 예입니다. 이 예에서는 필터링된 인덱스 FIBillOfMaterialsWithComponentID를 만든 다음 이를 SELECT 문에 대한 인덱스 힌트로 사용합니다. 필터링된 인덱스 조건자에는 ComponentID 533, 324 및 753의 데이터 행이 포함됩니다. 쿼리 조건자에도 ComponentID 533, 324 및 753의 데이터 행이 포함되지만 필터링된 인덱스에 없는 ComponentID 855 및 924도 포함하도록 결과 집합이 확장되어 있습니다 따라서 쿼리 최적화 프로그램에서는 필터링된 인덱스 힌트를 사용할 수 없으며 오류 8622가 발생합니다. 자세한 내용은 필터링된 인덱스 디자인 지침을 참조하십시오.

USE AdventureWorks2008R2;
GO
IF EXISTS (SELECT name FROM sys.indexes
    WHERE name = N'FIBillOfMaterialsWithComponentID' 
    AND object_id = OBJECT_ID(N'Production.BillOfMaterials'))
DROP INDEX FIBillOfMaterialsWithComponentID
    ON Production.BillOfMaterials;
GO
CREATE NONCLUSTERED INDEX "FIBillOfMaterialsWithComponentID"
    ON Production.BillOfMaterials (ComponentID, StartDate, EndDate)
    WHERE ComponentID IN (533, 324, 753);
GO
SELECT StartDate, ComponentID FROM Production.BillOfMaterials
    WITH( INDEX (FIBillOfMaterialsWithComponentID) )
    WHERE ComponentID in (533, 324, 753, 855, 924);
GO

쿼리 최적화 프로그램은 SET 옵션에 필터링된 인덱스에 대한 필수 값이 없으면 인덱스 힌트를 인식하지 않습니다. 자세한 내용은 CREATE INDEX(Transact-SQL)를 참조하십시오.

NOEXPAND 사용

NOEXPAND는 인덱싱된 뷰에만 적용됩니다. 인덱싱된 뷰란 고유한 클러스터형 인덱스가 만들어져 있는 뷰를 의미합니다. 인덱싱된 뷰와 기본 테이블 모두에 있는 열에 대한 참조가 포함된 쿼리의 경우 쿼리 최적화 프로그램이 쿼리를 실행하는 데 인덱싱된 뷰를 사용하는 것이 최상의 방법이라고 결정하면 뷰의 인덱스를 사용합니다. 이 기능을 인덱싱된 뷰 일치라고 하며 SQL Server Enterprise 및 Developer Edition에서만 지원됩니다.

단, 최적화 프로그램에서 일치시킬 인덱싱된 뷰를 고려하거나 NOEXPAND 힌트로 참조된 인덱싱된 뷰를 사용하려면 다음 SET 옵션을 ON으로 설정해야 합니다.

ANSI_NULLS

ANSI_WARNINGS

CONCAT_NULL_YIELDS_NULL

ANSI_PADDING

ARITHABORT1

QUOTED_IDENTIFIERS

1 ARITHABORT는 ANSI_WARNINGS가 ON으로 설정되어 있을 때 암시적으로 ON으로 설정됩니다. 따라서 이 설정을 수동으로 조정할 필요가 없습니다.

또한 NUMERIC_ROUNDABORT 옵션은 OFF로 설정해야 합니다.

최적화 프로그램이 인덱싱된 뷰에 대한 인덱스를 강제로 사용하게 하려면 NOEXPAND 옵션을 지정합니다. 이 힌트는 뷰가 쿼리에서도 명명되어 있는 경우에만 사용할 수 있습니다. SQL Server는 FROM 절에서 직접 뷰를 명명하지 않는 쿼리에서 특정 인덱싱된 뷰를 강제로 사용하도록 힌트를 제공하지 않지만 쿼리 최적화 프로그램은 쿼리에서 직접 참조되지 않은 경우에도 인덱싱된 뷰의 사용을 고려합니다.

자세한 내용은 뷰의 인덱스 확인을 참조하십시오.

테이블 힌트를 쿼리 힌트로 사용

OPTION (TABLE HINT) 절을 사용하여 테이블 힌트를 쿼리 힌트로 지정할 수도 있습니다. 테이블 힌트는 계획 지침의 컨텍스트에서 쿼리 힌트로만 사용하는 것이 좋습니다. 다른 임시 쿼리의 경우에는 이러한 힌트를 테이블 힌트로만 지정합니다. 자세한 내용은 쿼리 힌트(Transact-SQL)를 참조하십시오.

사용 권한

KEEPIDENTITY, IGNORE_CONSTRAINTS 및 IGNORE_TRIGGERS 힌트를 사용하려면 테이블에 대한 ALTER 권한이 필요합니다.

1. TABLOCK 힌트를 사용하여 잠금 방법 지정

다음 예에서는 Production.Product 테이블에 공유 잠금을 사용하고 UPDATE 문이 끝날 때까지 유지하도록 지정합니다.

USE AdventureWorks2008R2;
GO
UPDATE Production.Product
WITH (TABLOCK)
SET ListPrice = ListPrice * 1.10
WHERE ProductNumber LIKE 'BK-%';
GO

2. FORCESEEK 힌트를 사용하여 Index Seek 연산 지정

다음 예에서는 인덱스 지정 없이 FORCESEEK 힌트를 사용하여 쿼리 최적화 프로그램이 Sales.SalesOrderDetail 테이블에서 Index Seek 연산을 수행하도록 지정합니다.

USE AdventureWorks2008R2;
GO
SELECT *
FROM Sales.SalesOrderHeader AS h
INNER JOIN Sales.SalesOrderDetail AS d WITH (FORCESEEK)
    ON h.SalesOrderID = d.SalesOrderID 
WHERE h.TotalDue > 100
AND (d.OrderQty > 5 OR d.LineTotal < 1000.00);
GO

다음 예에서는 인덱스와 함께 FORCESEEK 힌트를 사용하여 쿼리 최적화 프로그램이 지정된 인덱스 및 인덱스 열에서 Index Seek 연산을 수행하도록 지정합니다.

USE AdventureWorks2008R2; 
GO
SELECT h.SalesOrderID, h.TotalDue, d.OrderQty
FROM Sales.SalesOrderHeader AS h
    INNER JOIN Sales.SalesOrderDetail AS d 
    WITH (FORCESEEK (PK_SalesOrderDetail_SalesOrderID_SalesOrderDetailID (SalesOrderID))) 
    ON h.SalesOrderID = d.SalesOrderID 
WHERE h.TotalDue > 100
AND (d.OrderQty > 5 OR d.LineTotal < 1000.00); 
GO

3. FORCESCAN 힌트를 사용하여 Index Scan 연산 지정

다음 예에서는 FORCESCAN 힌트를 사용하여 쿼리 최적화 프로그램이 Sales.SalesOrderDetail 테이블에서 검색 작업을 수행하도록 지정합니다.

USE AdventureWorks2008R2; 
GO
SELECT h.SalesOrderID, h.TotalDue, d.OrderQty
FROM Sales.SalesOrderHeader AS h
    INNER JOIN Sales.SalesOrderDetail AS d 
    WITH (FORCESCAN) 
    ON h.SalesOrderID = d.SalesOrderID 
WHERE h.TotalDue > 100
AND (d.OrderQty > 5 OR d.LineTotal < 1000.00);