Microsoft MVP성태의 닷넷 이야기
VC++: 104. std::call_once를 이용해 thread-safe한 Singleton 객체 생성 [링크 복사], [링크+제목 복사],
조회: 27906
글쓴 사람
정성태 (techsharer at outlook.com)
홈페이지
첨부 파일

(시리즈 글이 2개 있습니다.)
Windows: 13. InitOnceExecuteOnce API 소개
; https://www.sysnet.pe.kr/2/0/432

VC++: 104. std::call_once를 이용해 thread-safe한 Singleton 객체 생성
; https://www.sysnet.pe.kr/2/0/11091




std::call_once를 이용해 thread-safe한 Singleton 객체 생성

"thread-safe한 Singleton 객체 생성"에 대해 전에도 한번 글을 쓴 적이 있는데요.

C++에서 싱글톤 구현하기
; https://www.sysnet.pe.kr/2/0/846

마침 gamecodi에서 이와 관련된 글이 나오면서,

Double checked locking pattern is Fixed in C++11 
; http://www.gamecodi.com/board/zboard-id-GAMECODI_Talkdev-no-4347-z-16.htm
; http://preshing.com/20130930/double-checked-locking-is-fixed-in-cpp11/

Implementing a Thread-safe Singleton with C++11 
; http://www.nuonsoft.com/blog/2012/10/21/implementing-a-thread-safe-singleton-with-c11/comment-page-1/

C++ 11부터는 call_once로,

call_once Function
; https://docs.microsoft.com/en-us/cpp/standard-library/mutex-functions#call_once

안전하게 구현한다는 것을 알게 되었습니다. ^^ 그러니까, 다음과 같이 Singleton 클래스를 만들어 주면 됩니다.

#include "stdafx.h"
#include <mutex>

using namespace std;

class MySingleton
{
private:
    static MySingleton* _inst;
    static std::once_flag _flag1;

    static MySingleton* getInstance()
    {
        std::call_once(_flag1, 
            []() { _inst = new MySingleton(); });

        return _inst;
    }
};

memory barrier니, lock이니, volatile이니... 신경 쓸 필요 없이 딱 저 한 줄이면 됩니다. ^^




쓰다 보니, call_once가 왠지 예전에 소개했던 InitOnceExecuteOnce와 기능이 겹칩니다.

Vista & Longhorn: 13. InitOnceExecuteOnce API 소개
; https://www.sysnet.pe.kr/2/0/432

혹시, 윈도우에서는 call_once 내부적으로 InitOnceExecuteOnce Win32 API를 이용해 구현되지 않았을까요? ^^ 확인 방법은 다음과 같이 코드를 구성하고,

#include "stdafx.h"
#include <mutex>
#include <iostream>

using namespace std;

std::once_flag flag1;

void print_out()
{
    std::cout << "Simple example: called once\n";
    getchar();
}

int main()
{
    std::call_once(flag1, print_out);

    return 0;
}

print_out에 BP(Breakpoint)를 걸어두면 됩니다. Visual Studio 디버깅으로 콜 스택을 확인해 보니,

>    ConsoleApplication1.exe!print_out() Line 14 C++
    ConsoleApplication1.exe!std::_Invoker_functor::_Call<void (__cdecl&)(void)>(void(*)() _Obj) Line 1375 C++
    ConsoleApplication1.exe!std::invoke<void (__cdecl&)(void)>(void(*)() _Obj) Line 1443  C++
    ConsoleApplication1.exe!std::_Invoke_stored_explicit<void (__cdecl&)(void),std::exception_ptr &,0>(std::tuple<void (__cdecl&)(void),std::exception_ptr &> && _Tuple, std::integer_sequence<unsigned int,0> __formal) Line 470 C++
    ConsoleApplication1.exe!std::_Callback_once<std::tuple<void (__cdecl&)(void),std::exception_ptr &>,std::integer_sequence<unsigned int,0>,1>(void * __formal, void * _Pv, void * * __formal) Line 497  C++
    ntdll.dll!RtlRunOnceExecuteOnce()   Unknown
    KernelBase.dll!_InitOnceExecuteOnce@16()   Unknown
    msvcp140d.dll!__crtInitOnceExecuteOnce(_RTL_RUN_ONCE * InitOnce, int(__stdcall*)(_RTL_RUN_ONCE *, void *, void * *) InitFn, void * Parameter, void * * Context) Line 262    C++
    msvcp140d.dll!std::_Execute_once(std::once_flag & _Flag, int(__stdcall*)(void *, void *, void * *) _Lambda_fp, void * _Pv) Line 12  C++
    ConsoleApplication1.exe!std::call_once<void (__cdecl&)(void)>(std::once_flag & _Flag, void(*)() _Fx) Line 519 C++
    ConsoleApplication1.exe!main() Line 20  C++
    ConsoleApplication1.exe!invoke_main() Line 64   C++
    ConsoleApplication1.exe!__scrt_common_main_seh() Line 253   C++
    ConsoleApplication1.exe!__scrt_common_main() Line 296   C++
    ConsoleApplication1.exe!mainCRTStartup() Line 17    C++
    kernel32.dll!@BaseThreadInitThunk@12() Unknown
    ntdll.dll!__RtlUserThreadStart()    Unknown
    ntdll.dll!__RtlUserThreadStart@8() Unknown

오호~~~ 정말 ^^ InitOnceExecuteOnce를 호출하고 있습니다.

그렇다면, Vista 이전의 운영체제에서는 InitOnceExecuteOnce가 없는데 어떻게 구현이 되는 걸까요? 테스트를 위해 "Platform Toolset"을 "Visual Studio 2013 - Windows XP (v120_xp)"로 맞추고 빌드하면 print_out의 콜 스택이 다음과 같이 바뀝니다.

>    ConsoleApplication1.exe!print_out() Line 14 C++
    ConsoleApplication1.exe!std::_Bind<1,void,void (__cdecl*const)(void)>::_Do_call<>(std::tuple<> _Myfargs, std::_Arg_idx<> __formal) Line 1149    C++
    ConsoleApplication1.exe!std::_Bind<1,void,void (__cdecl*const)(void)>::operator()<>() Line 1138 C++
    ConsoleApplication1.exe!std::_Once_target<std::_Bind<1,void,void (__cdecl*const)(void)> >::_Call() Line 445 C++
    msvcp120d.dll!_Do_call(void * _Tgt) Line 76 C++
    msvcp120d.dll!_Call_onceEx(int * cntrl, void(*)(void *) func, void * arg) Line 42   C++
    ConsoleApplication1.exe!std::_Call_it<std::_Bind<1,void,void (__cdecl*const)(void)> >(std::once_flag & _Flag, std::_Bind<1,void,void (__cdecl*const)(void)> _Tgt) Line 473    C++
    ConsoleApplication1.exe!std::call_once<void (__cdecl&)(void)>(std::once_flag & _Flag, void(*)() _Fx) Line 482 C++
    ConsoleApplication1.exe!main() Line 20  C++
    ConsoleApplication1.exe!__tmainCRTStartup() Line 626    C
    ConsoleApplication1.exe!mainCRTStartup() Line 466   C
    kernel32.dll!@BaseThreadInitThunk@12() Unknown
    ntdll.dll!__RtlUserThreadStart()    Unknown
    ntdll.dll!__RtlUserThreadStart@8() Unknown

안 보이는 걸로 봐서는, msvcp120.dll 자체에서 구현하는 식으로 우회하는 것 같습니다.

(첨부한 파일은 이 글의 예제 코드를 포함합니다.)




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







[최초 등록일: ]
[최종 수정일: 4/5/2021]

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

비밀번호

댓글 작성자
 




... 151  152  153  154  155  156  157  158  159  [160]  161  162  163  164  165  ...
NoWriterDateCnt.TitleFile(s)
1109정성태8/26/201132575오류 유형: 135. 어느 순간 Active Directory 접속이 안되는 문제
1108정성태8/22/201132800오류 유형: 134. OLE/COM Object Viewer - DllRegisterServer in IVIEWERS.DLL failed. [1]
1107정성태8/21/201131941디버깅 기술: 43. Windows Form의 Load 이벤트에서 발생하는 예외가 Visual Studio에서 잡히지 않는 문제
1106정성태8/20/201130521웹: 26. FailedRequestTracing 설정으로 인한 iisexpress.exe 비정상 종료 문제
1105정성태8/19/201129627.NET Framework: 238. Web Site Model 프로젝트에서 Trace.WriteLine 출력이 dbgview.exe에서 확인이 안 되는 문제파일 다운로드1
1104정성태8/19/201130356웹: 25. WebDev보다 IIS Express가 더 나은 점 - 다중 가상 디렉터리 매핑 [1]
1103정성태8/19/201136403오류 유형: 133. WCF 포트 바인딩 실패 오류 - TCP error(10013) [1]
1102정성태8/19/201132987Math: 1. 방탈출3 - Room 10의 '중복가능한 조합' 문제를 위한 C# 프로그래밍 [2]파일 다운로드1
1101정성태8/19/201132806.NET Framework: 237. WCF AJAX 서비스와 JavaScript 간의 DateTime 연동 [1]파일 다운로드1
1100정성태8/17/201131846.NET Framework: 236. SqlDbType - DateTime, DateTime2, DateTimeOffset의 차이점파일 다운로드1
1099정성태8/15/201130797오류 유형: 132. 어느 순간 갑자기 접속이 안 되는 TFS 서버
1098정성태8/15/201153092웹: 24. 네이버는 어떻게 로그인 처리를 할까요? [2]
1097정성태8/15/201124191.NET Framework: 235. 메서드의 메타 데이터 토큰 값으로 클래스를 찾아내는 방법
1096정성태8/15/201128429디버깅 기술: 42. Watson Bucket 정보를 이용한 CLR 응용 프로그램 예외 분석 - (2)
1095정성태8/14/201128721디버깅 기술: 41. Windbg - 비정상 종료된 닷넷 프로그램의 StackTrace에서 보이는 offset 값 의미
1094정성태8/14/201133323오류 유형: 131. Fiddler가 강제 종료된 경우, 웹 사이트 방문이 안되는 현상
1093정성태7/27/201126929오류 유형: 130. Unable to connect to the Microsoft Visual Studio Remote Debugging Monitor ... Access is denied.
1092정성태7/22/201129540Team Foundation Server: 46. 코드 이외의 파일에 대해 소스 제어에서 제외시키는 방법
1091정성태7/21/201128418개발 환경 구성: 128. WP7 Emulator 실행 시 audiodg.exe의 CPU 소모율 증가 [2]
1089정성태7/18/201134065.NET Framework: 234. 왜? Button 컨트롤에는 MouseDown/MouseUp 이벤트가 발생하지 않을까요?파일 다운로드1
1088정성태7/16/201127429.NET Framework: 233. Entity Framework 4.1 - 윈도우 폰 7에서의 CodeFirst 순환 참조 문제파일 다운로드1
1087정성태7/15/201129723.NET Framework: 232. Entity Framework 4.1 - CodeFirst 개체의 직렬화 시 순환 참조 해결하는 방법 - 두 번째 이야기파일 다운로드1
1086정성태7/14/201130930.NET Framework: 231. Entity Framework 4.1 - CodeFirst 개체의 직렬화 시 순환 참조 해결하는 방법 [1]파일 다운로드1
1085정성태7/14/201131361.NET Framework: 230. Entity Framework 4.1 - Code First + WCF 서비스 시 EndpointNotFoundException 오류 - 두 번째 이야기파일 다운로드1
1084정성태7/11/201137099.NET Framework: 229. SQL 서버 - DB 테이블의 데이터 변경에 대한 알림 처리 [4]파일 다운로드1
1083정성태7/11/201131023.NET Framework: 228. Entity Framework 4.1 - Code First + WCF 서비스 시 EndpointNotFoundException 오류
... 151  152  153  154  155  156  157  158  159  [160]  161  162  163  164  165  ...