간단한 기교를 부려볼 겸 클래스의 멤버함수를 쓰레드 함수로 작성하는 방법을 배워 보도록 하겠습니다.
(간단히 쓰레드 사용법도 배우고 일석이조! 야호! )
아직까진 그런 적은 없지만, 쓰레드를 돌릴 때 간간히 멤버함수를 쓰레드 함수로 제작하고플 때가 있더군요.
혹, 그런 분들을 위해 알려 드리겠습니다.
class TestClass { INT B; VOID TestFunction( INT A ) { B = A }; //이 함수를 쓰레드로 돌리고자 합니다. VOID Start(); //이 함수에서 TestFunction을 호출하지요. }; |
Start() 멤버 함수에서 TestFunction()을 쓰레드로 돌리고자 한다고 해보죠.
VOID TestClass ::Start() {
HANDLE hThread = ( HANDLE ) __beginthread( NULL, 0, &TestFunction, NULL, NULL, NULL ); // ........ 무엇가의 작업을 하고. WaitForSingleObject( hThread, INFINITE ); CloseHandle( hThread ); } |
물론 안됩니다. 쓰레드로 돌아갈 함수는 정적으로 선언되어야 하며 정적 함수여야 하죠. 따라서,
class TestClass {
static VOID WINAPI TestFunction( INT A ); // 이제 원래 의도대로 이 함수를 정적 함수로 제작합니다. VOID Start(); }; |
역시 물론 안됩니다. 직접 닥쳐보시면 알겠지만, 우리가 돌리고자 했던 TestFunction 함수에서는 B라는 비정적 멤버를 참조하고 있습니다.
결국은 다른 방법을 찾아야 합니다.
class TestClass {
VOID TestFunction( INT A ); //다시 원래대로 돌리고 static VOID WINAPI TestFunctionThread( LPVOID ); //대신 쓰레드가 돌아갈 정적 함수를 만듭니다. VOID Start(); }; VOID Start() { HANDLE hThread = ( HANDLE ) __beginthread( NULL, 0, &TestFunctionThread, this, NULL, NULL ); // ........ 무엇가의 작업을 하고. WaitForSingleObject( hThread, INFINITE ); CloseHandle( hThread ); } VOID WINAPI TestFunctionThread( LPVOID p ) { ( (TestClass* ) p )->TestFunction( 3 ); } |
자, 위 예를 통해 해결했습니다. 포인터. 즉 동적 호출을 통해서 정적 멤버 함수를 호출하였습니다.
TestFunction의 인자 값도 동적으로 주고 싶다면,
struct Arg { TestClass* p; INT A; }; VOID Start() { Arg arg = { this, 3 }; HANDLE hThread = ( HANDLE ) __beginthread( NULL, 0, &TestFunctionThread, &arg, NULL, NULL ); // ........ 무엇가의 작업을 하고. WaitForSingleObject( hThread, INFINITE ); CloseHandle( hThread ); } VOID WINAPI TestFunctionThread( LPVOID p ) { Arg* pArg = ( Arg* ) p; pArg->p->TestFunction( p->A ); } |