C언어는 메모리와 문자열 처리에서 효율성과 유연성을 제공하지만, 유니코드 문자열 처리에서는 추가적인 기법과 이해가 필요합니다. 유니코드는 전 세계의 문자와 기호를 포함한 범용 문자 집합으로, 멀티바이트 및 와이드 문자열을 통해 표현됩니다. 본 기사에서는 C언어로 유니코드 문자열을 처리하는 방법을 기본 개념부터 실용적인 구현까지 자세히 다룹니다.
유니코드와 멀티바이트 문자열의 기본 이해
유니코드는 다양한 언어의 문자를 통합적으로 표현하기 위해 개발된 범용 문자 집합으로, UTF-8, UTF-16, UTF-32와 같은 여러 인코딩 방식으로 구현됩니다.
유니코드란 무엇인가
유니코드는 각각의 문자에 고유한 코드 포인트를 부여하며, 이는 모든 언어의 문자를 표현할 수 있는 단일 표준입니다. 예를 들어, ‘A’의 코드 포인트는 U+0041입니다.
C언어에서의 멀티바이트 문자열
C언어는 기본적으로 ASCII 문자열을 다루지만, 유니코드와 같은 멀티바이트 문자열도 지원합니다. 멀티바이트 문자열은 한 문자가 여러 바이트로 표현될 수 있는 문자열로, UTF-8 인코딩 방식에서 흔히 사용됩니다.
멀티바이트와 와이드 문자열의 차이
- 멀티바이트 문자열: char 배열로 표현되며, UTF-8 인코딩을 주로 사용합니다.
- 와이드 문자열: wchar_t 배열로 표현되며, UTF-16 또는 UTF-32 인코딩을 주로 사용합니다.
C언어에서 유니코드 문자열을 올바르게 처리하려면 이러한 개념과 인코딩 방식을 정확히 이해하는 것이 중요합니다.
wchar_t와 UTF-8 문자열 비교
C언어에서 유니코드 문자열을 처리할 때 대표적으로 사용되는 두 가지 표현 방식은 wchar_t와 UTF-8 문자열입니다. 이 두 가지는 각각의 장단점과 특징을 가지고 있습니다.
wchar_t 문자열
wchar_t는 C언어에서 와이드 문자열을 표현하기 위한 데이터 타입으로, UTF-16이나 UTF-32와 같은 고정 길이 인코딩을 주로 사용합니다.
장점
- 각 문자가 고정된 크기를 가지므로 처리 속도가 일정합니다.
- 대부분의 문자가 한 요소로 표현되기 때문에 접근과 조작이 간단합니다.
단점
- 메모리 사용량이 크며, 특히 UTF-32는 필요 이상으로 많은 공간을 차지할 수 있습니다.
- 플랫폼에 따라 wchar_t의 크기가 달라질 수 있어 이식성이 떨어질 수 있습니다.
UTF-8 문자열
UTF-8은 멀티바이트 인코딩 방식으로, 문자의 길이가 가변적입니다. UTF-8은 기본적으로 char 배열로 표현됩니다.
장점
- ASCII와 호환되므로 기존 시스템과의 통합이 용이합니다.
- 메모리 효율성이 높아 영문과 숫자를 주로 사용하는 경우 메모리 절약이 가능합니다.
- 플랫폼 간 일관성이 뛰어납니다.
단점
- 각 문자의 길이가 가변적이므로 특정 위치로의 접근이 복잡합니다.
- 문자열 길이 계산 및 조작 시 추가 작업이 필요합니다.
실용적인 선택
- wchar_t: 고정된 크기의 유니코드 처리와 빠른 접근이 필요한 경우 적합합니다.
- UTF-8: 메모리 효율성과 ASCII 호환성이 중요한 경우 적합합니다.
코드 예시
#include <stdio.h>
#include <wchar.h>
int main() {
// wchar_t 문자열
wchar_t wide_string[] = L"유니코드 테스트";
wprintf(L"와이드 문자열: %ls\n", wide_string);
// UTF-8 문자열
char utf8_string[] = "Unicode Test";
printf("UTF-8 문자열: %s\n", utf8_string);
return 0;
}
이처럼 사용 사례에 따라 wchar_t와 UTF-8 문자열을 선택해 유니코드 데이터를 효과적으로 처리할 수 있습니다.
유니코드 문자열 입력 및 출력
C언어에서 유니코드 문자열을 처리하려면, 입력 및 출력 함수와 함께 유니코드 인코딩 방식에 맞는 데이터를 다루는 것이 중요합니다. 이 섹션에서는 wchar_t와 UTF-8 문자열의 입력 및 출력 방법을 살펴봅니다.
wchar_t 문자열 입력 및 출력
wchar_t 문자열은 와이드 문자 지원이 필요한 라이브러리와 함수로 처리됩니다.
#include <stdio.h>
#include <wchar.h>
#include <locale.h>
int main() {
setlocale(LC_ALL, ""); // 로케일 설정
wchar_t input[100];
wprintf(L"와이드 문자열 입력: ");
fgetws(input, 100, stdin); // 와이드 문자열 입력
wprintf(L"입력된 문자열: %ls\n", input); // 와이드 문자열 출력
return 0;
}
주요 함수
fgetws
: 표준 입력에서 와이드 문자열 읽기.wprintf
: 와이드 문자열 출력.
UTF-8 문자열 입력 및 출력
UTF-8 문자열은 일반 char 배열로 처리되며, 표준 C 함수로도 충분히 다룰 수 있습니다.
#include <stdio.h>
int main() {
char input[100];
printf("UTF-8 문자열 입력: ");
fgets(input, 100, stdin); // UTF-8 문자열 입력
printf("입력된 문자열: %s\n", input); // UTF-8 문자열 출력
return 0;
}
주요 함수
fgets
: 표준 입력에서 문자열 읽기.printf
: 문자열 출력.
와이드 문자열과 UTF-8 문자열 간의 변환
유니코드 문자열을 변환하기 위해 mbstowcs
(멀티바이트 → 와이드)와 wcstombs
(와이드 → 멀티바이트)를 사용할 수 있습니다.
#include <stdio.h>
#include <stdlib.h>
#include <wchar.h>
int main() {
char utf8_string[] = "Hello, Unicode!";
wchar_t wide_string[100];
// UTF-8 → wchar_t 변환
mbstowcs(wide_string, utf8_string, 100);
wprintf(L"와이드 문자열: %ls\n", wide_string);
// wchar_t → UTF-8 변환
char converted_utf8[100];
wcstombs(converted_utf8, wide_string, 100);
printf("UTF-8 문자열: %s\n", converted_utf8);
return 0;
}
정리
- wchar_t: 와이드 입력 및 출력을 위해
wprintf
,fgetws
등 와이드 함수 사용. - UTF-8: 일반 입력 및 출력 함수인
fgets
,printf
로 충분히 처리 가능. - 인코딩 변환:
mbstowcs
,wcstombs
로 UTF-8과 wchar_t 간의 변환 지원.
이처럼 적절한 함수와 기술을 활용하면 C언어에서 유니코드 문자열 입력과 출력을 쉽게 구현할 수 있습니다.
유니코드 문자열 변환
C언어에서 유니코드 문자열은 다양한 인코딩 방식으로 표현될 수 있습니다. UTF-8, UTF-16, UTF-32 간의 변환은 유니코드 데이터를 효율적으로 처리하고 저장하기 위해 필수적인 작업입니다. 이 섹션에서는 유니코드 문자열 변환 방법과 그 실용적인 활용 사례를 다룹니다.
UTF-8에서 UTF-16 또는 UTF-32로 변환
UTF-8 문자열을 UTF-16 또는 UTF-32로 변환하려면 mbstowcs
와 같은 표준 라이브러리 함수를 사용할 수 있습니다.
#include <stdio.h>
#include <stdlib.h>
#include <wchar.h>
int main() {
char utf8_string[] = "안녕하세요";
wchar_t wide_string[100];
// UTF-8 → UTF-16/UTF-32 (와이드 문자열) 변환
mbstowcs(wide_string, utf8_string, 100);
wprintf(L"와이드 문자열: %ls\n", wide_string);
return 0;
}
UTF-16 또는 UTF-32에서 UTF-8로 변환
반대로 UTF-16/UTF-32 문자열을 UTF-8로 변환할 때는 wcstombs
를 사용합니다.
#include <stdio.h>
#include <stdlib.h>
#include <wchar.h>
int main() {
wchar_t wide_string[] = L"안녕하세요";
char utf8_string[100];
// UTF-16/UTF-32 → UTF-8 변환
wcstombs(utf8_string, wide_string, 100);
printf("UTF-8 문자열: %s\n", utf8_string);
return 0;
}
외부 라이브러리 활용
C 표준 라이브러리는 기본적인 변환 기능만 제공하므로, 복잡한 변환 작업에는 ICU(International Components for Unicode)와 같은 외부 라이브러리를 사용할 수 있습니다.
ICU를 활용한 UTF 변환
ICU 라이브러리는 u_strToUTF8
과 같은 함수를 제공하여 효율적인 UTF 변환을 지원합니다.
#include <unicode/ucnv.h>
#include <unicode/ustring.h>
void convertUTF16ToUTF8(const UChar *utf16, char *utf8, int32_t utf8Capacity) {
UErrorCode status = U_ZERO_ERROR;
u_strToUTF8(utf8, utf8Capacity, NULL, utf16, -1, &status);
if (U_FAILURE(status)) {
printf("변환 실패: %s\n", u_errorName(status));
}
}
응용 사례
- 네트워크 전송: UTF-8 문자열은 압축된 형식으로 데이터 전송에 적합합니다.
- 파일 저장: UTF-16은 텍스트 파일의 다국어 지원을 위해 자주 사용됩니다.
- 문자열 비교: 변환된 데이터를 기반으로 다양한 언어에서 문자열 정렬과 비교가 가능합니다.
정리
- UTF-8, UTF-16, UTF-32 간 변환은
mbstowcs
와wcstombs
로 기본적으로 처리 가능. - 복잡한 변환 작업은 ICU 라이브러리와 같은 외부 도구를 활용.
- 변환 작업은 다양한 응용 프로그램에서 유니코드 데이터의 상호운용성을 보장.
유니코드 변환은 문자열 데이터를 효과적으로 저장, 처리, 전송하는 데 중요한 역할을 합니다.
표준 라이브러리와 유니코드 지원
C언어의 표준 라이브러리는 유니코드 문자열 처리를 위한 기본적인 기능을 제공합니다. 이를 통해 와이드 문자열 및 멀티바이트 문자열을 효율적으로 다룰 수 있습니다.
wchar.h 헤더의 유니코드 지원
wchar.h
는 와이드 문자열과 문자를 처리하기 위한 함수를 제공합니다.
주요 함수
wprintf
: 와이드 문자열 출력.wscanf
: 와이드 문자열 입력.wcscpy
: 와이드 문자열 복사.wcscmp
: 와이드 문자열 비교.wcslen
: 와이드 문자열 길이 계산.
코드 예시
#include <wchar.h>
#include <locale.h>
int main() {
setlocale(LC_ALL, ""); // 로케일 설정
wchar_t str1[] = L"안녕하세요";
wchar_t str2[100];
wcscpy(str2, str1); // 문자열 복사
wprintf(L"복사된 문자열: %ls\n", str2);
wprintf(L"문자열 길이: %zu\n", wcslen(str1)); // 문자열 길이 계산
return 0;
}
stdlib.h 헤더의 멀티바이트 지원
stdlib.h
는 멀티바이트와 와이드 문자열 간 변환을 지원합니다.
주요 함수
mbstowcs
: 멀티바이트 문자열 → 와이드 문자열 변환.wcstombs
: 와이드 문자열 → 멀티바이트 문자열 변환.
코드 예시
#include <stdlib.h>
#include <stdio.h>
#include <wchar.h>
int main() {
char mbstr[] = "Hello, World!";
wchar_t wcstr[100];
mbstowcs(wcstr, mbstr, 100); // 멀티바이트 → 와이드 변환
wprintf(L"와이드 문자열: %ls\n", wcstr);
char converted_mbstr[100];
wcstombs(converted_mbstr, wcstr, 100); // 와이드 → 멀티바이트 변환
printf("멀티바이트 문자열: %s\n", converted_mbstr);
return 0;
}
locale.h 헤더와 로케일 설정
유니코드 문자열 처리는 로케일 설정에 따라 달라질 수 있으므로, locale.h
를 사용해 올바른 로케일을 설정해야 합니다.
#include <locale.h>
setlocale(LC_ALL, ""); // 현재 시스템 로케일 설정
한계와 보완
- 표준 라이브러리 한계: 기본적인 문자열 변환과 조작은 가능하지만, 복잡한 유니코드 기능(예: 대소문자 변환, 정렬, 정규화 등)은 지원하지 않습니다.
- 외부 라이브러리 보완: ICU, libunistring 등을 활용해 고급 유니코드 기능을 보완할 수 있습니다.
정리
wchar.h
와stdlib.h
는 와이드 및 멀티바이트 문자열 처리를 지원.- 로케일 설정은 유니코드 데이터의 적절한 처리를 위해 필수적.
- 복잡한 유니코드 처리에는 표준 라이브러리를 보완하는 외부 도구를 활용.
표준 라이브러리를 활용하면 기본적인 유니코드 처리가 가능하며, 외부 도구를 통해 더 높은 수준의 기능을 구현할 수 있습니다.
외부 라이브러리를 활용한 유니코드 처리
C언어의 표준 라이브러리는 유니코드 문자열 처리를 위한 기본적인 도구를 제공하지만, 복잡한 작업이나 고급 기능을 구현하기에는 한계가 있습니다. 이를 보완하기 위해 다양한 외부 라이브러리가 제공됩니다. 이 섹션에서는 대표적인 유니코드 지원 라이브러리와 그 활용 방법을 살펴봅니다.
ICU(International Components for Unicode)
ICU는 유니코드와 관련된 고급 작업(정규화, 정렬, 텍스트 분석 등)을 지원하는 강력한 라이브러리입니다.
ICU의 주요 기능
- 유니코드 정규화: 문자열의 일관된 표현을 보장.
- 대소문자 변환: 다양한 언어에 적합한 대소문자 변환.
- 텍스트 정렬: 유니코드 규칙에 따른 문자열 정렬.
ICU 사용 예제
#include <unicode/ustring.h>
#include <unicode/ucnv.h>
#include <unicode/unorm2.h>
#include <stdio.h>
int main() {
UErrorCode status = U_ZERO_ERROR;
const UNormalizer2 *norm = unorm2_getInstance(NULL, "nfc", UNORM2_COMPOSE, &status);
if (U_FAILURE(status)) {
printf("ICU 초기화 실패: %s\n", u_errorName(status));
return 1;
}
UChar source[] = {0x1100, 0x1161, 0}; // '가'를 조합형으로 표현
UChar result[100];
int32_t len = unorm2_normalize(norm, source, -1, result, 100, &status);
if (U_SUCCESS(status)) {
printf("정규화 성공, 길이: %d\n", len);
} else {
printf("정규화 실패: %s\n", u_errorName(status));
}
return 0;
}
libunistring
libunistring은 텍스트 처리에 필요한 유니코드 함수와 데이터 구조를 제공합니다.
- UTF-8, UTF-16, UTF-32 변환.
- 문자열 대소문자 변환 및 검색.
- 텍스트 조작 함수.
libunistring 사용 예제
#include <unistr.h>
#include <stdio.h>
int main() {
const uint8_t utf8[] = "Hello, 유니코드!";
size_t length;
uint32_t *utf32 = u8_to_u32(utf8, strlen((char *)utf8), NULL, &length);
if (utf32) {
printf("UTF-8 → UTF-32 변환 성공, 길이: %zu\n", length);
free(utf32);
} else {
printf("변환 실패\n");
}
return 0;
}
왜 외부 라이브러리를 사용할까?
- 고급 기능: 표준 라이브러리로는 불가능한 정규화, 텍스트 분석 등 제공.
- 다국어 지원: 다양한 언어와 문화권에 맞는 텍스트 처리를 지원.
- 효율성: 복잡한 유니코드 처리 작업을 최적화.
외부 라이브러리 선택 기준
- 프로젝트 요구사항에 적합한 기능 제공 여부.
- 사용하기 쉬운 API 및 문서화.
- 성능 및 메모리 효율성.
정리
ICU와 libunistring 같은 외부 라이브러리는 유니코드 데이터를 효율적으로 처리하기 위한 강력한 도구를 제공합니다. 이러한 라이브러리는 표준 라이브러리를 보완하며, 특히 다국어 환경에서 유용하게 사용됩니다.
유니코드 문자열 정렬과 검색
유니코드 문자열을 정렬하고 검색하는 작업은 국제화와 다국어 지원이 필요한 소프트웨어에서 중요한 역할을 합니다. C언어에서는 기본적인 비교 함수와 외부 라이브러리를 활용해 유니코드 문자열 정렬과 검색을 구현할 수 있습니다.
유니코드 문자열 비교
문자열을 정렬하거나 검색하기 위해서는 우선 문자열 비교가 필요합니다. 기본적으로 C언어는 wcscmp
함수로 와이드 문자열을 비교할 수 있습니다.
코드 예시: 와이드 문자열 비교
#include <wchar.h>
#include <locale.h>
int main() {
setlocale(LC_ALL, "");
wchar_t str1[] = L"안녕하세요";
wchar_t str2[] = L"안녕히 가세요";
int result = wcscmp(str1, str2);
if (result < 0) {
wprintf(L"'%ls'이(가) '%ls'보다 작습니다.\n", str1, str2);
} else if (result > 0) {
wprintf(L"'%ls'이(가) '%ls'보다 큽니다.\n", str1, str2);
} else {
wprintf(L"'%ls'과(와) '%ls'이(가) 같습니다.\n", str1, str2);
}
return 0;
}
유니코드 문자열 정렬
여러 문자열을 정렬하려면 qsort
함수와 사용자 정의 비교 함수를 사용할 수 있습니다.
코드 예시: 와이드 문자열 정렬
#include <wchar.h>
#include <stdio.h>
#include <stdlib.h>
#include <locale.h>
int compare_wstrings(const void *a, const void *b) {
return wcscmp(*(const wchar_t **)a, *(const wchar_t **)b);
}
int main() {
setlocale(LC_ALL, "");
wchar_t *strings[] = {
L"안녕하세요",
L"Hello",
L"こんにちは",
L"你好"
};
size_t count = sizeof(strings) / sizeof(strings[0]);
qsort(strings, count, sizeof(wchar_t *), compare_wstrings);
wprintf(L"정렬된 문자열:\n");
for (size_t i = 0; i < count; ++i) {
wprintf(L"%ls\n", strings[i]);
}
return 0;
}
유니코드 문자열 검색
문자열에서 특정 서브스트링을 검색하려면 wcsstr
함수를 사용할 수 있습니다.
코드 예시: 와이드 문자열 검색
#include <wchar.h>
#include <locale.h>
int main() {
setlocale(LC_ALL, "");
wchar_t str[] = L"안녕하세요, 유니코드 세계!";
wchar_t target[] = L"유니코드";
wchar_t *result = wcsstr(str, target);
if (result) {
wprintf(L"'%ls'에서 '%ls' 발견!\n", str, target);
} else {
wprintf(L"'%ls'에서 '%ls'을(를) 찾을 수 없습니다.\n", str, target);
}
return 0;
}
외부 라이브러리를 활용한 정렬
ICU는 유니코드 규칙에 따른 문자열 정렬을 제공합니다.
#include <unicode/ucol.h>
#include <unicode/ustring.h>
#include <stdio.h>
void sortStrings(const UChar *strings[], int count) {
UErrorCode status = U_ZERO_ERROR;
UCollator *collator = ucol_open("en_US", &status);
if (U_SUCCESS(status)) {
for (int i = 0; i < count - 1; ++i) {
for (int j = i + 1; j < count; ++j) {
if (ucol_strcoll(collator, strings[i], -1, strings[j], -1) > 0) {
const UChar *temp = strings[i];
strings[i] = strings[j];
strings[j] = temp;
}
}
}
ucol_close(collator);
}
}
정리
- 표준 함수:
wcscmp
,wcsstr
등을 활용해 기본적인 비교와 검색 가능. - 정렬:
qsort
를 사용한 사용자 정의 비교 함수로 문자열 정렬 구현. - 외부 라이브러리: ICU 같은 라이브러리로 국제화 규칙에 따른 정렬과 검색 지원.
적절한 방법을 선택하여 유니코드 문자열 데이터를 효율적으로 정렬하고 검색할 수 있습니다.
디버깅과 문제 해결
C언어에서 유니코드 문자열을 처리할 때는 다양한 문제가 발생할 수 있습니다. 이러한 문제를 신속하게 식별하고 해결하는 것은 프로젝트의 성공을 위해 필수적입니다. 이 섹션에서는 일반적으로 발생하는 문제와 그 해결 방법을 소개합니다.
문제 1: 인코딩 불일치
UTF-8, UTF-16, UTF-32 등 서로 다른 인코딩 방식이 혼합되어 발생하는 문제입니다.
- 현상: 출력이 깨지거나 예상치 못한 결과가 나타남.
- 원인: 문자열의 인코딩 방식과 처리 방식이 일치하지 않음.
해결 방법
- 문자열의 인코딩 방식을 명확히 정의.
- 필요한 경우 변환 함수(
mbstowcs
,wcstombs
)를 사용해 동일한 인코딩으로 변환. - 디버깅 단계에서 문자열의 바이트 값을 출력해 인코딩을 확인.
#include <stdio.h>
void print_bytes(const char *str) {
while (*str) {
printf("%02x ", (unsigned char)*str);
str++;
}
printf("\n");
}
문제 2: 로케일 설정 누락
유니코드 데이터를 처리할 때 올바른 로케일이 설정되지 않은 경우 문제가 발생할 수 있습니다.
- 현상: 와이드 문자열 출력이 깨지거나, 입력이 제대로 처리되지 않음.
- 원인:
setlocale
함수가 호출되지 않음.
해결 방법
프로그램 시작 시 적절한 로케일을 설정.
#include <locale.h>
setlocale(LC_ALL, "");
문제 3: 메모리 부족 또는 초과
유니코드 문자열 변환 시 필요한 메모리를 정확히 계산하지 않아 문제가 발생.
- 현상: 프로그램이 크래시되거나 예기치 않은 동작.
- 원인: 변환 시 배열 크기를 초과하거나 부족한 메모리를 할당.
해결 방법
- 변환 전 필요한 메모리를 정확히 계산.
- 동적 메모리 할당(
malloc
)을 사용할 때 할당 크기를 명시.
#include <stdlib.h>
size_t required_size = mbstowcs(NULL, utf8_str, 0) + 1; // 널 문자 포함
wchar_t *wide_str = malloc(required_size * sizeof(wchar_t));
문제 4: 디버깅 어려움
유니코드 문자열은 눈에 보이는 값과 내부 데이터가 다르기 때문에 디버깅이 어려울 수 있습니다.
- 현상: 문자열 출력이 예상과 다르거나 결과가 불명확함.
해결 방법
- 문자열의 코드 포인트를 출력해 각 문자의 유니코드 값을 확인.
- 인코딩 변환 결과를 바이너리 값으로 출력.
코드 예시: 유니코드 코드 포인트 출력
#include <wchar.h>
void print_codepoints(const wchar_t *str) {
while (*str) {
wprintf(L"U+%04x ", *str);
str++;
}
wprintf(L"\n");
}
문제 5: 외부 라이브러리 사용 문제
ICU와 같은 외부 라이브러리를 사용하는 경우 초기화나 사용 방법의 잘못으로 문제가 발생.
- 현상: 라이브러리 함수 호출 실패 또는 잘못된 결과.
- 원인: 라이브러리 초기화 코드 누락 또는 잘못된 API 호출.
해결 방법
- 라이브러리 문서를 참조해 올바르게 초기화.
UErrorCode
등 상태 값을 사용해 오류를 점검.- 테스트 코드 작성으로 예상 결과와 비교.
정리
- 인코딩 문제: 문자열 인코딩 확인 및 변환으로 해결.
- 로케일 문제:
setlocale
로 적절한 로케일 설정. - 메모리 문제: 정확한 크기 계산과 동적 메모리 할당.
- 디버깅 도구: 바이너리 값과 코드 포인트 출력.
- 외부 라이브러리: 오류 상태 점검 및 문서 참고.
이러한 접근 방법을 통해 유니코드 문자열 처리 중 발생하는 문제를 효과적으로 해결할 수 있습니다.
요약
C언어에서 유니코드 문자열을 다루는 방법은 인코딩 방식의 이해, 변환, 정렬, 검색, 디버깅에 이르기까지 다양합니다. UTF-8, UTF-16, UTF-32 간 변환과 표준 라이브러리 및 외부 도구를 활용한 처리는 유니코드 문자열의 정확하고 효율적인 관리에 필수적입니다. 이를 통해 다국어 환경에서 안정적이고 강력한 소프트웨어를 개발할 수 있습니다.