Microsoft MVP성태의 닷넷 이야기
VC++: 130. C++ string의 c_str과 data 함수의 차이점 [링크 복사], [링크+제목 복사]
조회: 1714
글쓴 사람
홈페이지
첨부 파일
 

C++ string의 c_str과 data 함수의 차이점

이에 대해서는 다음의 글에 잘 정리가 되어 있습니다.

string c_str() vs. data()
; https://stackoverflow.com/questions/194634/string-c-str-vs-data

어쨌든 공식적으로는 C++03 이전에는 구현체마다 다를 수 있었지만 C++11부터는 단일하게 null 종료 문자를 가진 유형으로 반환한다고 합니다.

In C++11 onwards, both functions are required to be the same. i.e. data is now required to be null-terminated. According to cppreference: "The returned array is null-terminated, that is, data() and c_str() perform the same function."


그러던 것이 C++17 표준부터는 data()는 "const"를 떼는 것으로 바뀝니다. 그래서 C++11 이전에는 다음의 코드가 문제없이 컴파일되었지만,와 함께,

const char* p1 = txt.data();
const char* p2 = txt.c_str();

"ISO C++17 Standard (/std:c++17)"부터는 data가 const 유형으로 반환하지 않으므로 다음과 같이 바꿔야 합니다.같은 표현도 가능합니다.

// 컴파일 에러 발생 - Error C2440 'initializing': cannot convert from 'const _Elem *' to 'char *'
// const char* p1 = txt.data();

char* p1 = txt.data(); // char* 변수로도 반환 가능
const char* p2 = txt.c_str();




참고로, C++03 이전에는 달랐다고 하는데 구현체마다 그럴 가능성은 있겠지만 대부분은 string이 문자열에 주로 사용되었으므로 동일하게 바뀐 것 같습니다. 예를 들어 Linux g++에서 "C++03 (-std=c++03)" 옵션으로 data(), c_str()의 코드를 보면,

const _CharT*
data() const _GLIBCXX_NOEXCEPT
{ return _M_data(); }

const _CharT*
c_str() const _GLIBCXX_NOEXCEPT
{ return _M_data(); }

동일한 _M_data() 함수를 호출하기 때문에 다른 동작을 할 여지가 없습니다. 혹시나 싶어 Visual Studio 2015의 C++ 프로젝트에서 "Platform target" 설정을 "Visual Studio 2015 - Windows XP (v140_xp)"로 한 경우의 std::string 소스 코드도 살펴봤는데요, 다만 이번에는 다음과 같이 차이점을 설명하는 (오래된) 주석은 확인할 수 있었습니다.

const _Elem *data() const _NOEXCEPT
    {   // return pointer to nonmutable array
    return (this->_Myptr());
    }

const _Elem *c_str() const _NOEXCEPT
    {   // return pointer to null-terminated nonmutable array
    return (this->_Myptr());
    }

그래도 코드 자체는 _Myptr() 함수를 부르는 것으로 동일하게 처리하지만. ^^;




[이 글에 대해서 여러분들과 의견을 공유하고 싶습니다. 틀리거나 미흡한 부분 또는 의문 사항이 있으시면 언제든 댓글 남겨주십시오.]

[연관 글]





[최초 등록일: ]
[최종 수정일: 5/20/2019 ]

Creative Commons License
이 저작물은 크리에이티브 커먼즈 코리아 저작자표시-비영리-변경금지 2.0 대한민국 라이센스에 따라 이용하실 수 있습니다.
by SeongTae Jeong, mailto:techsharer@outlook.com

비밀번호

댓글 쓴 사람
 



2019-05-20 04시37분
[kernel] // 컴파일 에러 발생 - Error C2440 'initializing': cannot convert from 'const _Elem *' to 'char *'
를 발생시킨 소스를 공유해주실 수 있나요? 저 에러메시지는 const* 인 data를 char*에 대입할 때 발생하는 메시지 같은데요...
[손님]
2019-05-20 04시40분
[kernel] 주석처리한 라인이 https://onlinegdb.com/rk9cY0yaE 와 vs2019 /std:c++17 에서 컴파일 에러를 만들지 않았습니다.
[손님]
2019-05-20 11시39분
아... 제가 헷갈렸습니다. 맞습니다. C++17과 그 하위의 버전을 번갈아가면서 테스트하다가 그렇게 꼬였습니다. ^^; 말씀하신 것처럼 C++14 이하 버전에서 const*로 반환하는데 그걸 char* 변수로 받다가 발생한 문제였습니다.
정성태

1  2  3  4  5  6  7  [8]  9  10  11  12  13  14  15  ...
NoWriterDateCnt.TitleFile(s)
12107정성태1/10/2020560.NET Framework: 877. C# - 프로세스의 모든 핸들을 열람 - 두 번째 이야기
12106정성태1/8/2020611VC++: 136. C++ - OSR Driver Loader와 같은 Legacy 커널 드라이버 설치 프로그램 제작 [1]
12105정성태1/8/2020510디버깅 기술: 153. C# - PEB를 조작해 로드된 DLL을 숨기는 방법
12104정성태1/9/2020851DDK: 9. 커널 메모리를 읽고 쓰는 NT Legacy driver와 C# 클라이언트 프로그램 [3]
12103정성태4/23/20201523DDK: 8. Visual Studio 2019 + WDK Legacy Driver 제작- Hello World 예제 [1]파일 다운로드2
12102정성태1/6/2020601디버깅 기술: 152. User 권한(Ring 3)의 프로그램에서 _ETHREAD 주소(및 커널 메모리를 읽을 수 있다면 _EPROCESS 주소) 구하는 방법
12101정성태1/8/2020620.NET Framework: 876. C# - PEB(Process Environment Block)를 통해 로드된 모듈 목록 열람
12100정성태1/3/2020429.NET Framework: 875. .NET 3.5 이하에서 IntPtr.Add 사용
12099정성태1/3/2020612디버깅 기술: 151. Windows 10 - Process Explorer로 확인한 Handle 정보를 windbg에서 조회
12098정성태1/2/2020578.NET Framework: 874. C# - 커널 구조체의 Offset 값을 하드 코딩하지 않고 사용하는 방법
12097정성태1/2/2020459디버깅 기술: 150. windbg - Wow64, x86, x64에서의 커널 구조체(예: TEB) 구조체 확인
12096정성태1/2/2020643디버깅 기술: 149. C# - DbgEng.dll을 이용한 간단한 디버거 제작
12095정성태12/27/2019756VC++: 135. C++ - string_view의 동작 방식
12094정성태12/26/2019654.NET Framework: 873. C# - 코드를 통해 PDB 심벌 파일 다운로드 방법
12093정성태12/26/2019807.NET Framework: 872. C# - 로딩된 Native DLL의 export 함수 목록 출력파일 다운로드1
12092정성태12/25/2019678디버깅 기술: 148. cdb.exe를 이용해 (ntdll.dll 등에 정의된) 커널 구조체 출력하는 방법
12091정성태12/25/2019900디버깅 기술: 147. pdb 파일을 다운로드하기 위한 symchk.exe 실행에 필요한 최소 파일 [1]
12090정성태12/24/2019642.NET Framework: 871. .NET AnyCPU로 빌드된 PE 헤더의 로딩 전/후 차이점
12089정성태12/23/2019541디버깅 기술: 146. gflags와 _CrtIsMemoryBlock을 이용한 Heap 메모리 손상 여부 체크
12088정성태12/23/2019442Linux: 28. Linux - 윈도우의 "Run as different user" 기능을 shell에서 실행하는 방법
12087정성태12/21/2019576디버깅 기술: 145. windbg/sos - Dictionary의 entries 배열 내용을 모두 덤프하는 방법 (do_hashtable.py)
12086정성태12/20/2019696디버깅 기술: 144. windbg - Marshal.FreeHGlobal에서 발생한 덤프 분석 사례
12085정성태12/20/2019569오류 유형: 586. iisreset - The data is invalid. (2147942413, 8007000d) 오류 발생 - 두 번째 이야기 [1]
12084정성태12/21/2019650디버깅 기술: 143. windbg/sos - Hashtable의 buckets 배열 내용을 모두 덤프하는 방법 (do_hashtable.py)
12083정성태12/17/2019949Linux: 27. linux - lldb를 이용한 .NET Core 응용 프로그램의 메모리 덤프 분석 방법 [2]
12082정성태12/17/2019697오류 유형: 585. lsof: WARNING: can't stat() fuse.gvfsd-fuse file system
1  2  3  4  5  6  7  [8]  9  10  11  12  13  14  15  ...