Microsoft MVP성태의 닷넷 이야기
COM 개체 관련: 19. COM의 Apartment를 이해해 보자. [링크 복사], [링크+제목 복사],
조회: 22322
글쓴 사람
정성태 (techsharer at outlook.com)
홈페이지
첨부 파일
 

마침, COM 관련한 자료를 보고 나니 생각이 나는군요.

이번 토픽은 "문제"와 함께 풀어나가는 식으로 구성을 해보겠습니다.(음... 이런 식으로 토픽 쓰는 시간을 벌 수도 있군요. ^^;)
참고로, 이 문제를 푸실 수 있는 자격은, 저에게 최초로 ^^ "ATL 강의"를 들었던 "L" 업체분들을 제외하고 모두 가능합니다.

제 사이트에 오시는 분들이 대부분 .NET만을 하시는 분들임을 감안하면, 거의 답변이 안 달릴 것 같은데... ^^;



[Q1] COM의 Apartment에 대해서 많이 들어보셨을 것입니다. STA, MTA가 대표적인 예이지요.
그럼, 다음과 같은 STA에서 생성된 STA COM 개체를 전역 변수에 보관했다가 다른 STA 스레드에서 호출하면 어떨까요? 이 코드는 "예외 없이" 동작할까요?

환경:	SimpleObject COM 개체 - STA
		Console Application

ISimpleObjectPtr g_ptr;

int _tmain(int , char** )
{
	// STA 스레드에서 STA COM 개체를 생성
	CoInitializeEx( NULL, COINIT_APARTMENTTHREADED );
	{
		g_ptr = ISimpleObjectPtr( __uuidof( SimpleObject ) );
		g_ptr->MyMethod();
		
		DWORD dwAnotherThreadID;
		HANDLE hHandle = ::CreateThread( NULL, 0, AnotherThread, 0, 0, &dwAnotherThreadID );
		CloseHandle( hHandle );		
	}
	
	getchar();
	g_ptr = 0;
	CoUninitialize();
}

DWORD WINAPI AnotherThread( LPVOID lpParameter )
{
	// STA 스레드에서 전역 변수로 존재하는 STA COM 개체를 직접 호출
	CoInitializeEx( NULL, COINIT_APARTMENTTHREADED );
	g_ptr->MyMethod();
	CoUninitialize();

	return 0;
}



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







[최초 등록일: ]
[최종 수정일: 8/19/2021]

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

비밀번호

댓글 작성자
 



2006-10-10 11시33분
저도 문제에 도전해보고 싶은데, C++ 을 거의 몰라서... ^_^;;; 혹시 Thread Local Storage (TLS) 관련 문제인가요?
songgun
2006-10-10 11시58분
흔히들, "STA 개체는 스레드 간에 호출하려면 인터페이스 포인터를 마샬링해서 전달해야 된다고 알고 있습니다." 바로 이 문구가 정확한지 않은지 곱씹어 보는 의미에서 질문을 드리는 것입니다. ^^
kevin25
2006-10-12 09시49분
[텨텨텨] 예외란게 Exception을 말씀하시는 것인지 "항상" 이란 뜻인지 모르겠습니다만...

위 코드 조각만 봐서는 일단 '호출'을 하는 데는 아무런 문제가 없죠.
다만 다중 쓰레드의 동기화 관점에서 보면 오류 가능성을 가지고 있겠지요?
STA의 sematics 상 MyMethod()는 단일 쓰레드에 의해 호출된다고 스스로 가정할 수 있지만
위 코드는 동시에 두 쓰레드가 MyMethod() 호출이 가능한 상황이 연출 됩니다.
(마샬링 된 인터페이스 포인터를 전역 변수에 기록해 두어야만 오류 수정이 가능합니다.)

이거 너무 어려운 토픽을 잡으신듯....
저도 몇번 STA, MTA 토픽에 덤비려다 설명할게 너무 많아 포기했었다는...
(누구게? -_-;)
텨~텨~텨~~~
[guest]
2006-10-13 08시00분
유수석님... ^^ 예. 맞습니다. 제가 원하던 답이었습니다.
왜 이런 문제를 냈냐면요... ^^; "STA 개체는 스레드 간에 마샬링이 필요하다"라는 책의 내용만을 읽어온 분들이 "호출"조차도 안 되는 것으로 알고 있어서 직접 확인해 보시라고 내본 것입니다. 역시나 말씀하신 것처럼 동기화에 문제가 발생하지만요... ^^

솔직히 저도 사실 아직도 STA/MTA에 대한 메카니즘을 완전히 이해하지 못 해서 완벽하게 쓸 예정은 아닙니다. 단지 내부 수준까지는 아니더라도 "실용적인" 수준으로만 써보려고 합니다. ^^
kevin25
2006-10-13 08시24분
문제랑은 전현 상관없지만... ^_^;;;

'텨텨텨' 라는 아이디 보고 유경상님이라는 감이 바로 왔습니다.
아마 유경상님 블로그에 자주 가는 사람들은 말투만 보고서도 바로 눈치 채실듯...
songgun
2006-10-13 09시16분
[유경상] 단박에 알아내시다니... 너무 티를 냈나 보군요... -_-;
너무 버릇없이 글을 쓰지 않았나 걱정도 되고...
(실명을 밝히지 않고 장난한 거 죄송합니다... 너그러이 용서 해주세요... ^^)
[guest]
2006-10-13 10시11분
에이... ^^ 유수석님이라면 언제든지, 어떤 식으로든지 환영이옵니다.
kevin25
2015-10-21 12시38분
What is COM marshaling and how do I use it?
; https://devblogs.microsoft.com/oldnewthing/20151020-00/?p=91321

What are the rules for CoMarshalInterThreadInterfaceInStream and CoGetInterfaceAndReleaseStream?
; https://devblogs.microsoft.com/oldnewthing/20151021-00/?p=91311

What are the rules for CoMarshalInterface and CoUnmarshalInterface?
; https://devblogs.microsoft.com/oldnewthing/20151022-00/?p=91301

CoGetInterfaceAndReleaseStream does not mix with smart pointers
; https://devblogs.microsoft.com/oldnewthing/20151023-00/?p=91291
정성태

... 31  32  33  34  35  [36]  37  38  39  40  41  42  43  44  45  ...
NoWriterDateCnt.TitleFile(s)
13041정성태4/28/202214591개발 환경 구성: 642. Informix 데이터베이스 docker 환경 구성
13040정성태4/27/202214221VC++: 156. 비주얼 스튜디오 - Linux C/C++ 프로젝트에서 openssl 링크하는 방법
13039정성태4/27/202217342.NET Framework: 1999. C# - Playwright를 이용한 간단한 브라우저 제어 실습
13038정성태4/26/202213513오류 유형: 806. twine 실행 시 ConfigParser.ParsingError: File contains parsing errors: /root/.pypirc
13037정성태4/25/202214302.NET Framework: 1998. Azure Functions를 사용한 간단한 실습
13036정성태4/24/202215247.NET Framework: 1997. C# - nano 시간을 가져오는 방법 [2]
13035정성태4/22/202216504Windows: 204. Windows 10부터 바뀐 QueryPerformanceFrequency, QueryPerformanceCounter
13034정성태4/21/202214054.NET Framework: 1996. C# XingAPI - 주식 종목에 따른 PBR, PER, ROE, ROA 구하는 방법(t3320, t8430 예제)파일 다운로드1
13033정성태4/18/202215195.NET Framework: 1195. C# - Thread.Yield와 Thread.Sleep(0)의 차이점(?)
13032정성태4/17/202215373오류 유형: 805. Github의 50MB 파일 크기 제한 - warning: GH001: Large files detected. You may want to try Git Large File Storage
13031정성태4/15/202215083.NET Framework: 1194. C# - IdealProcessor와 ProcessorAffinity의 차이점
13030정성태4/15/202213588오류 유형: 804. 정규 표현식 오류 - Quantifier {x,y} following nothing.
13029정성태4/14/202215018Windows: 203. iisreset 후에도 이전에 설정한 전역 환경 변수가 w3wp.exe에 적용되는 문제
13028정성태4/13/202215145.NET Framework: 1193. (appsettings.json처럼) web.config의 Debug/Release에 따른 설정 적용
13027정성태4/12/202215012.NET Framework: 1192. C# - 환경 변수의 변화를 알리는 WM_SETTINGCHANGE Win32 메시지 사용법파일 다운로드1
13026정성태4/11/202216357.NET Framework: 1191. C 언어로 작성된 FFmpeg Examples의 C# 포팅 전체 소스 코드 [3]
13025정성태4/11/202215632.NET Framework: 1190. C# - ffmpeg(FFmpeg.AutoGen)를 이용한 vaapi_encode.c, vaapi_transcode.c 예제 포팅
13024정성태4/7/202213873.NET Framework: 1189. C# - 런타임 환경에 따라 달라진 AppDomain.GetCurrentThreadId 메서드
13023정성태4/6/202214465.NET Framework: 1188. C# - ffmpeg(FFmpeg.AutoGen)를 이용한 transcoding.c 예제 포팅 [3]
13022정성태3/31/202214021Windows: 202. 윈도우 11 업그레이드 - "PC Health Check"를 통과했지만 여전히 업그레이드가 안 되는 경우 해결책
13021정성태3/31/202215844Windows: 201. Windows - INF 파일을 이용한 장치 제거 방법
13020정성태3/30/202214090.NET Framework: 1187. RDP 접속 시 WPF UserControl의 Unloaded 이벤트 발생파일 다운로드1
13019정성태3/30/202214551.NET Framework: 1186. Win32 Message를 Code로부터 메시지 이름 자체를 텍스트로 구하고 싶다면?파일 다운로드1
13018정성태3/29/202215293.NET Framework: 1185. C# - Unsafe.AsPointer가 반환한 포인터는 pinning 상태일까요? [5]
13017정성태3/28/202214258.NET Framework: 1184. C# - GC Heap에 위치한 참조 개체의 주소를 알아내는 방법 - 두 번째 이야기 [3]
13016정성태3/27/202215964.NET Framework: 1183. C# 11에 추가된 ref 필드의 (우회) 구현 방법파일 다운로드1
... 31  32  33  34  35  [36]  37  38  39  40  41  42  43  44  45  ...