setjmp로.. 수동적인(?) 콜백 구조를 능동적(?)으로 바꿔보려 했는데.. 오류가 납니다..
슬아
질문 제목 :
setjmp로.. 수동적인(?)콜백 구조를 능동적(?)으로 바꿔보려 했는데.. 오류가 납니다..
질문 내용 :
클래스가 포함되어서 c게시판에는 어울리지 않을 수도 있지만..
setjmp 관련 질문은c게시판에나 어울릴듯 해서 여기다 그냥 올립니다;
그냥 문득 생각이 나서 한번 시도해 봤는데.. (그냥 실험삼아서 한것입니다;;)오류가 나더군요;
아래 빨간색으로 표시해 둔 부분에서 액세스 위반이 납니다.
구체적으로.. 첫 nextchild호출 후 메시지박스는 잘 뜹니다.
그런데.. 두번째 nextchild를 호출할때.. 빨간색으로 표시한 부분에서 오류가 납니다..
visual c++ ide에서 보여주는 콜스택을 보면 최상단이 00000000 번지로 되어 있었습니다..
그리고 00000000번지를 읽을 수 없다는 오류였습니다.
물론 c++까지 와서setjmp를 사용하는건 비평받을만한 행동이겠지만
실제로 사용하고자 하는건 아니고.. 그냥 실험삼아서 하는 거니 이해해 주세요;;;
혹시 제가 뭔가 잘못한 부분이 있을까요? 답변 부탁드립니다. 감사합니다.
class cenumchildwindows
{
private:
hwnd m_hparent; //이 윈도우의 차일드들을 열거합니다.
hwnd m_hgotchild; // enumproc에서 얻어낸 차일드 핸들을 이 변수에 담아서, nextchild에서 반환할 수 있도록 합니다.
bool m_fenumerating; // 현재열거가 진행중인지를 나타냅니다.
jmp_buf m_jmpbuf1;// nextchild메소드 내부에서 setjmp하는 jmp_buf입니다. (enumproc에서 이것에 대해 longjmp함)
jmp_buf m_jmpbuf2;// enumproc함수 내부에서 setjmp하는 jmp_buf입니다. (nextchild에서 이것에 대해 longjmp함)
private:
static bool callback enumproc(hwnd hwnd, lparam lparam)
{
cenumchildwindows *self = (cenumchildwindows*)lparam;
switch (setjmp(self-m_jmpbuf2))
{
case 0: // nextchild가 값을 반환할 수 있도록m_hgotchild에 차일드 핸들을넣고 nextchild함수로longjmp합니다.
self-m_hgotchild = hwnd;
longjmp(self-m_jmpbuf1, 1);
break;
case 1: //nextchild가 두번째 ~ n번째 호출되면, 이곳에 도달하고, 여기서 아무 일도 하지 않고true를반환하므로enumproc로의 다음 호출이 옵니다.
break;
case 2: // stop메소드나 cenumchildwindows의 소멸자로 인해 열거가 중단되었습니다. false를 반환하여 열거를 멈춥니다.
return false;
}
return true;
}
public:
cenumchildwindows(hwnd hparent)
: m_hparent(hparent)
, m_hgotchild(null)
, m_fenumerating(false)
{
}
~cenumchildwindows()
{
stop();
}
//다음 차일드윈도우 핸들을얻습니다. 끝까지 다 열거했으면 null을 반환합니다.
hwnd nextchild()
{
if (!m_fenumerating)
{
if (setjmp(m_jmpbuf1) == 0)
{
m_fenumerating = true;
enumchildwindows(m_hparent, &enumproc, (lparam)this);
m_fenumerating = false;
return null;
}
}
else
{
if (setjmp(m_jmpbuf1) == 0)
longjmp(m_jmpbuf2, 1); //access violation!
}
return m_hgotchild;
}
void stop()
{
if (m_fenumerating)
longjmp(m_jmpbuf2, 2);
}
};
// cenumchildwindows를 실제로 사용하는 코드
cenumchildwindows enumaaa(hwnd);
while (enumaaa.nextchild()) // nextchild는, 더이상 차일드가 없으면 null을 반환.
messagebox(hwnd, lasdf, lasdf, mb_ok);