기본 콘텐츠로 건너뛰기

GNU의 std::string은 같은 문자열을 공유한다.

보통 std::map에서 key로 std::string을 많이 쓰는데, 쓸 때마다 부담스러웠다. 이유는 key는 std::map에서 node를 만들 때 instance를 새로 만들고 복제하기 때문이다. 그러나 GNU의 std::string은 동일한 문자열은 같은 메모리영역을 공유하도록 만들었다. 따라서 복제가 이뤄져도 실제로 데이터를 복제하는 것이 아니라 내부 레퍼런스 카운터만 올라간다. /usr/include/c++/x.x.x/bits/basic_string.[h|tcc] 파일을 까보면 _M_is_shared()라는 private method가 assign을 비롯하여 여기저기 쓰여 있는 걸로 보아 확실하다. 이를 나름 증명해보자면, 아래 소스를 컴파일해서 돌려보자.

#include <iostream>
#include <string>
#include <map>
using namespace std;

map<string,void*> gMap;

int
main(int, char**)
{
    string str1("Hello, world!");
    string str2(str1);
    string str3; str3 = str1;
    string str4; str4.assign(str1);
    gMap.insert(map<string,void*>::value_type(str1, NULL));

    cout << "str1: " << (void*)str1.c_str() << endl;
    cout << "str2: " << (void*)str2.c_str() << endl;
    cout << "str3: " << (void*)str3.c_str() << endl;
    cout << "str4: " << (void*)str4.c_str() << endl;
    cout << "gMap.begin(): "
        << (void*)gMap.begin()->first.c_str() << endl;
    return 0;
}

역시나 char 배열을 어떻게 저장하느냐에 따라 c_str 구현이 달라지겠지만, 역시나 까봤더니 걍 포인터 리턴하는 것 말곤 없다. 여기서 위 값들이 동일하게 나온다면, 문자열 공간을 공유하는 것이라 하겠다.
$g++ -o shstr shstr.cpp
$./shstr
str1: 0x9be6014
str2: 0x9be6014
str3: 0x9be6014
str4: 0x9be6014
gMap.begin():
0x9be6014
참고로 실행환경은 libstdc++-3.4.4-2, libstdc++-4.1.2-12 모두 동일했다. str1~4가 모두 동일한 메모리 주소를 뱉는다는 것으로 실제로 복제가 일어나진 않는다는 것을 알았다. 소스를 좀더 까보면 데이터 변경이 일어날 때야 비로소 레퍼런스 카운트 떨구고 새로 메모리 주소를 할당 받아서 복제한 뒤 연산하는 것을 알 수 있다.


덧글:: object님 말씀대로 정확히 같은 문자열을 공유하는 것이 아니다. 복사와 대입을 할 때, CopyOnWrite철학으로 memory allocation과 memory copy(string copy)를 늦추는 행위를 할 뿐이다.

댓글

이 블로그의 인기 게시물

버즈 라이브 배터리 교체

나는 버즈 라이브(SM-R180)가 좋은데, 평가가 별루였는지, 해당 스타일로 버즈를 더 이상 만들지 않고 있다. 아무튼, 오래 쓴 버즈 라이브 배터리가 슬슬 맛이 가기 시작해서, 블로그 를 참조하면서 분해 및 교체를 하였다. (진짜 쉬움) 요로코롬 위아래를 살짝 눌러주면 뚜껑이 벌어진다. 안쪽 플라스틱은 오른쪽은 분홍색, 왼쪽은 회색이다. 리본 케이블 살짝 들어내고, 기판을 떼어내면, 작은 나사가 있다. 나사를 풀고, 플라스틱을 걷어내면, 검은 양면 테이프로 고정된 CR1254 배터리가 보인다. 잘 쑤셔서(?) 꺼낸다. 새로운 CR1254 배터리를 넣는다. 음극이 아래로 가도록 하고, 분해의 역순으로 조립하면 된다. 조립할 때, 아까 풀었던 나사는 잊지 말고 꼭 조여준다. (까먹고 조립해서 다시 뜯고 조립함) 충전도 잘 되고, 소리도 잘 나는거 보면, 조립도 잘 된 것 같다. 이렇게 버즈 라이브의 수명을 강제로 늘렸다. 나중에 본체 배터리도 갈아야겠다.

Windows 에서 절전을 깨우는 장치 찾기

참조:  https://www.reddit.com/r/computer/comments/wquswv/windows_11_pc_wakes_up_every_time_i_move_usb/ powercfg /devicequery wake_armed powercfg /deviceenablewake "[DEVICE]" # $PROFILE function Get-WakeArmedDevices { $devices = powercfg -devicequery wake_armed if ($devices) { $devices | ForEach-Object { $_.Trim() } } else { Write-Host "No devices are currently armed for wake events." } } function Set-EnableWakeOnDevice { param( [string]$deviceName ) sudo powercfg -deviceenablewake $deviceName } function Set-DisableWakeOnDevice { param( [string]$deviceName ) sudo powercfg -devicedisablewake $deviceName }