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

... 106  107  108  109  110  111  112  113  114  [115]  116  117  118  119  120  ...
NoWriterDateCnt.TitleFile(s)
11049정성태9/24/201620995오류 유형: 357. 윈도우 백업 시 오류 - 0x81000037
11048정성태9/24/201622023VC++: 100. 전역 변수 유형별 실행 파일 크기 차이점
11047정성태9/21/201625820기타: 61. algospot.com - 양자화(Quantization) 문제 [2]파일 다운로드1
11046정성태9/15/201627487개발 환경 구성: 298. Windows 10 - bash 실행 시 시작 디렉터리 자동 변경
11045정성태9/15/201620137Windows: 119. Windows 10 - bash 명령어 창을 실행했는데 바로 닫히는 경우
11044정성태9/15/201620388VS.NET IDE: 112. Visual Studio 확장 - 편집 화면 내에서 링크를 누르면 외부 웹 브라우저에서 열기
11043정성태9/15/201621812.NET Framework: 606. .NET 스레드 콜 스택 덤프 (7) - ClrMD(Microsoft.Diagnostics.Runtime)를 이용한 방법 [1]파일 다운로드1
11042정성태9/14/201619977오류 유형: 356. Unknown custom metadata item kind: 6
11041정성태9/10/201619454.NET Framework: 605. CLR4 보안 - yield 구문 내에서 SecurityCritical 메서드 사용 불가 - 2번째 이야기
11040정성태9/10/201626747.NET Framework: 604. C# Windows Forms - Drag & Drop 예제 코드 [2]파일 다운로드1
11039정성태9/9/201623229오류 유형: 355. Visual Studio 빌드 오류 - error CS0122: '__ComObject' is inaccessible due to its protection level
11038정성태9/9/201625086VC++: 99. 서로 다른 프로세스에서 WM_DROPFILES 메시지를 전송하는 방법파일 다운로드1
11037정성태9/8/201628314.NET Framework: 603. socket - shutdown 호출이 필요한 사례파일 다운로드1
11036정성태8/29/201624788개발 환경 구성: 297. 소스 코드가 없는 닷넷 어셈블리를 디버깅할 때 지역 변숫값을 확인하는 방법
11035정성태8/29/201620423오류 유형: 354. .NET Reflector - PDB 생성 화면에서 "Clear Store"를 하면 "Index and length must refer to a location within the string" 예외 발생
11034정성태8/25/201624439개발 환경 구성: 296. .NET Core 프로젝트를 NuGet Gallery에 배포하는 방법 [2]
11033정성태8/24/201622344오류 유형: 353. coreclr 빌드 시 error C3249: illegal statement or sub-expression for 'constexpr' function
11032정성태8/23/201621561개발 환경 구성: 295. 최신의 Visual C++ 컴파일러 도구를 사용하는 방법 [1]
11031정성태8/23/201617804오류 유형: 352. Error encountered while pushing to the remote repository: Response status code does not indicate success: 403 (Forbidden).
11030정성태8/23/201620344VS.NET IDE: 111. Team Explorer - 추가한 Git Remote 저장소가 Branch에 보이지 않는 경우
11029정성태8/18/201627477.NET Framework: 602. Process.Start의 cmd.exe에서 stdin만 redirect 하는 방법 [1]파일 다운로드1
11028정성태8/15/201621526오류 유형: 351. Octave 설치 시 JRE 경로 문제
11027정성태8/15/201622599.NET Framework: 601. ElementHost 컨트롤의 메모리 누수 현상
11026정성태8/13/201623578Math: 19. 행렬 연산으로 본 해밍코드
11025정성태8/12/201622296개발 환경 구성: 294. .NET Core 프로젝트에서 "Copy to Output Directory" 처리 [1]
11024정성태8/12/201621614오류 유형: 350. "nProtect GameMon" 실행 중에는 Visual Studio 디버깅이 안됩니다! [1]
... 106  107  108  109  110  111  112  113  114  [115]  116  117  118  119  120  ...