Namespaces
Variants

strcpy, strcpy_s

From cppreference.net
< c ‎ | string ‎ | byte
헤더 파일에 정의됨 <string.h>
(1)
char * strcpy ( char * dest, const char * src ) ;
(C99 이전)
char * strcpy ( char * restrict dest, const char * restrict src ) ;
(C99 이후)
errno_t strcpy_s ( char * restrict dest, rsize_t destsz, const char * restrict src ) ;
(2) (C11 이후)
1) src 가 가리키는 널 종료 바이트 문자열을(널 종결자 포함) dest 가 첫 번째 요소를 가리키는 문자 배열에 복사합니다.
동작은 dest 배열이 충분히 크지 않을 경우 정의되지 않습니다. 문자열이 겹칠 경우 동작은 정의되지 않습니다. 동작은 dest 가 문자 배열에 대한 포인터가 아니거나 src 가 null-terminated byte string을 가리키는 포인터가 아닐 경우 정의되지 않습니다.
2) (1) 과 동일하지만, 대상 배열의 나머지 부분을 지정되지 않은 값으로 덮어쓸 수 있으며, 다음 오류들이 런타임에 감지되어 현재 설치된 constraint handler 함수를 호출한다는 점이 다릅니다:
  • src 또는 dest 가 null 포인터인 경우
  • destsz 가 0이거나 RSIZE_MAX 보다 큰 경우
  • destsz strnlen_s ( src, destsz ) 보다 작거나 같은 경우; 즉, 잘림(truncation)이 발생할 경우
  • 소스 문자열과 대상 문자열 사이에 중복(overlap)이 발생하는 경우
동작은 다음 조건에서 정의되지 않습니다: dest <= strnlen_s ( src, destsz ) < destsz ; 다시 말해, destsz 의 잘못된 값은 버퍼 오버플로를 허용할 수 있습니다.
모든 경계 검사 함수와 마찬가지로, strcpy_s 는 구현체가 __STDC_LIB_EXT1__ 를 정의하고, 사용자가 __STDC_WANT_LIB_EXT1__ 를 정수 상수 1 으로 정의한 경우에만 사용 가능함이 보장됩니다. (이 정의는 <string.h> 포함 전에 이루어져야 합니다)

목차

매개변수

dest - 문자 배열에 대한 포인터
src - 복사할 null 종료 바이트 문자열에 대한 포인터
destsz - 최대 작성 문자 수, 일반적으로 대상 버퍼의 크기

반환값

1) dest 의 복사본을 반환합니다
2) 성공 시 0을 반환하고, 오류 시 0이 아닌 값을 반환합니다. 또한 오류 발생 시, dest [ 0 ] 에 0을 기록합니다 (단, dest 가 null 포인터이거나 destsz 가 0이거나 RSIZE_MAX 보다 큰 경우는 제외).

참고 사항

strcpy_s 는 효율성 향상을 위해 마지막으로 기록된 문자부터 destsz 까지 대상 배열을 덮어쓸 수 있습니다: 멀티바이트 블록으로 복사한 후 널 바이트를 확인할 수 있습니다.

함수 strcpy_s 는 BSD 함수 strlcpy 와 유사하지만, 다음 차이점이 있습니다

  • strlcpy 소스 문자열을 대상에 맞게 잘라냅니다 (이는 보안 위험 요소임)
  • strlcpy strcpy_s 가 수행하는 모든 런타임 검사를 수행하지 않습니다
  • strlcpy 는 호출이 실패할 경우 대상을 널 문자열로 설정하거나 핸들러를 호출하여 실패를 명확히 나타내지 않습니다.

비록 strcpy_s 가 잠재적인 보안 위험으로 인해 잘림을 금지하지만, 경계 검사가 포함된 strncpy_s 를 사용하여 문자열을 잘라낼 수 있습니다.

예제

#define __STDC_WANT_LIB_EXT1__ 1
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
int main(void)
{
    const char* src = "Take the test.";
//  src[0] = 'M' ; // 이는 정의되지 않은 동작입니다
    char dst[strlen(src) + 1]; // 널 종결자를 수용하기 위해 +1
    strcpy(dst, src);
    dst[0] = 'M'; // 정상 동작
    printf("src = %s\ndst = %s\n", src, dst);
#ifdef __STDC_LIB_EXT1__
    set_constraint_handler_s(ignore_handler_s);
    int r = strcpy_s(dst, sizeof dst, src);
    printf("dst = \"%s\", r = %d\n", dst, r);
    r = strcpy_s(dst, sizeof dst, "Take even more tests.");
    printf("dst = \"%s\", r = %d\n", dst, r);
#endif
}

가능한 출력:

src = Take the test.
dst = Make the test.
dst = "Take the test.", r = 0
dst = "", r = 22

참고문헌

  • C23 표준 (ISO/IEC 9899:2024):
  • 7.24.2.3 strcpy 함수 (p: TBD)
  • K.3.7.1.3 strcpy_s 함수 (p: TBD)
  • C17 표준 (ISO/IEC 9899:2018):
  • 7.24.2.3 strcpy 함수 (p: 264-265)
  • K.3.7.1.3 strcpy_s 함수 (p: 447)
  • C11 표준 (ISO/IEC 9899:2011):
  • 7.24.2.3 strcpy 함수 (p: 363)
  • K.3.7.1.3 strcpy_s 함수 (p: 615-616)
  • C99 표준 (ISO/IEC 9899:1999):
  • 7.21.2.3 strcpy 함수 (p: 326)
  • C89/C90 표준 (ISO/IEC 9899:1990):
  • 4.11.2.3 strcpy 함수

참고 항목

특정 길이의 문자를 한 문자열에서 다른 문자열로 복사합니다
(함수)
한 버퍼를 다른 버퍼로 복사합니다
(함수)
(C95) (C11)
와이드 문자열을 다른 와이드 문자열로 복사합니다
(함수)
(dynamic memory TR)
문자열의 복사본을 할당합니다
(함수)