Microsoft MVP성태의 닷넷 이야기
COM 개체 관련: 19. COM의 Apartment를 이해해 보자. [링크 복사], [링크+제목 복사],
조회: 22265
글쓴 사람
정성태 (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
정성태

1  2  3  4  5  6  [7]  8  9  10  11  12  13  14  15  ...
NoWriterDateCnt.TitleFile(s)
13768정성태10/15/20245411C/C++: 179. C++ - _O_WTEXT, _O_U16TEXT, _O_U8TEXT의 Unicode stream 모드파일 다운로드2
13767정성태10/14/20244784오류 유형: 929. bpftrace 수행 시 "ERROR: Could not resolve symbol: /proc/self/exe:BEGIN_trigger"
13766정성태10/14/20244562C/C++: 178. C++ - 파일에 대한 Text 모드의 "translated" 동작파일 다운로드1
13765정성태10/12/20245271오류 유형: 928. go build 시 "package maps is not in GOROOT" 오류
13764정성태10/11/20245640Linux: 85. Ubuntu - 원하는 golang 버전 설치
13763정성태10/11/20244988Linux: 84. WSL / Ubuntu 20.04 - bpftool 설치
13762정성태10/11/20245018Linux: 83. WSL / Ubuntu 22.04 - bpftool 설치
13761정성태10/11/20244917오류 유형: 927. WSL / Ubuntu - /usr/include/linux/types.h:5:10: fatal error: 'asm/types.h' file not found
13760정성태10/11/20245462Linux: 82. Ubuntu - clang 최신(stable) 버전 설치
13759정성태10/10/20246373C/C++: 177. C++ - 자유 함수(free function) 및 주소 지정 가능한 함수(addressable function) [6]
13758정성태10/8/20245580오류 유형: 926. dotnet tools를 sudo로 실행하는 경우 command not found
13757정성태10/8/20245538닷넷: 2306. Linux - dotnet tool의 설치 디렉터리가 PATH 환경변수에 자동 등록이 되는 이유
13756정성태10/8/20245635오류 유형: 925. ssh로 docker 접근을 할 때 "... malformed HTTP status code ..." 오류 발생
13755정성태10/7/20246031닷넷: 2305. C# 13 - (9) 메서드 바인딩의 우선순위를 지정하는 OverloadResolutionPriority 특성 도입 (Overload resolution priority)파일 다운로드1
13754정성태10/4/20245583닷넷: 2304. C# 13 - (8) 부분 메서드 정의를 속성 및 인덱서에도 확대파일 다운로드1
13753정성태10/4/20245596Linux: 81. Linux - PATH 환경변수의 적용 규칙
13752정성태10/2/20246299닷넷: 2303. C# 13 - (7) ref struct의 interface 상속 및 제네릭 제약으로 사용 가능 [6]파일 다운로드1
13751정성태10/2/20245414C/C++: 176. C/C++ - ARM64로 포팅할 때 유의할 점
13750정성태10/1/20245295C/C++: 175. C++ - WinMain/wWinMain 호출 전의 CRT 초기화 단계
13749정성태9/30/20245547닷넷: 2302. C# - ssh-keygen으로 생성한 Private Key와 Public Key 연동파일 다운로드1
13748정성태9/29/20245747닷넷: 2301. C# - BigInteger 타입이 byte 배열로 직렬화하는 방식
13747정성태9/28/20245599닷넷: 2300. C# - OpenSSH의 공개키 파일에 대한 "BEGIN OPENSSH PUBLIC KEY" / "END OPENSSH PUBLIC KEY" PEM 포맷파일 다운로드1
13746정성태9/28/20245697오류 유형: 924. Python - LocalProtocolError("Illegal header value ...")
13745정성태9/28/20245561Linux: 80. 리눅스 - 실행 중인 프로세스 내부의 환경변수 설정을 구하는 방법 (lldb)
13744정성태9/27/20245999닷넷: 2299. C# - Windows Hello 사용자 인증 다이얼로그 표시하기파일 다운로드1
13743정성태9/26/20246437닷넷: 2298. C# - Console 프로젝트에서의 await 대상으로 Main 스레드 활용하는 방법 [1]
1  2  3  4  5  6  [7]  8  9  10  11  12  13  14  15  ...