1

DFS 연결 구성 요소 레이블을 쓰고 있는데, 기본 아이디어는 정말 간단합니다. DFS를 네 번째 이웃 (왼쪽, 오른쪽, 위, 아래) 반복적으로 적용하는 것입니다.깊이 우선 스택 오버플로 검색

문제는

0xC00000FD: Stack overflow (: 0x00000001, 0x001D2EB4) 

나는 그것이 너무 깊이 간다 있기 때문에 생각, 연결된 영역이 너무 큰 경우, 말, 100 * 100 픽셀, 그것은 런타임 오류를 얻을 수 있다는 것입니다. 어떤 최적화 또는 솔루션이 있습니까? 여기

코드이다

void DFS_Traversal(cv::Mat &InputMat, cv::Mat &LabelMat, cv::Point2i cur_SP, int Thresh, int cur_Label){ 

    if (cur_SP.y > 2 && cur_SP.y < (InputMat.rows - 2) && cur_SP.x > 2 && cur_SP.x < (InputMat.cols - 2)){ 
     uchar* pre_Input_rowPtr = InputMat.ptr<uchar>(cur_SP.y - 1); 
     uchar* cur_Input_rowPtr = InputMat.ptr<uchar>(cur_SP.y); 
     uchar* next_Input_rowPtr = InputMat.ptr<uchar>(cur_SP.y + 1); 
     uchar* pre_Label_rowPtr = LabelMat.ptr<uchar>(cur_SP.y - 1); 
     uchar* cur_Label_rowPtr = LabelMat.ptr<uchar>(cur_SP.y); 
     uchar* next_Label_rowPtr = LabelMat.ptr<uchar>(cur_SP.y + 1); 

     //cur_Label_rowPtr[cur_SP.x] = cur_Label; 

     //Left Point 
     if (cur_Label_rowPtr[cur_SP.x - 1] == 0 && std::abs(cur_Input_rowPtr[cur_SP.x] - cur_Input_rowPtr[cur_SP.x - 1]) < Thresh){ 
      cv::Point2i left_Point(cur_SP.x - 1, cur_SP.y); 

      cur_Label_rowPtr[cur_SP.x - 1] = cur_Label; 
      DFS_Traversal(InputMat, LabelMat, left_Point, Thresh, cur_Label); 
     } 
     //Right Point 
     if (cur_Label_rowPtr[cur_SP.x + 1] == 0 && std::abs(cur_Input_rowPtr[cur_SP.x] - cur_Input_rowPtr[cur_SP.x + 1]) < Thresh){ 
      cv::Point2i right_Point(cur_SP.x + 1, cur_SP.y); 

      cur_Label_rowPtr[cur_SP.x + 1] = cur_Label; 
      DFS_Traversal(InputMat, LabelMat, right_Point, Thresh, cur_Label); 
     } 
     //Up Point 
     if (pre_Label_rowPtr[cur_SP.x] == 0 && std::abs(cur_Input_rowPtr[cur_SP.x] - pre_Input_rowPtr[cur_SP.x]) < Thresh){ 
      cv::Point2i up_Point(cur_SP.x, cur_SP.y - 1); 

      pre_Label_rowPtr[cur_SP.x] = cur_Label; 
      DFS_Traversal(InputMat, LabelMat, up_Point, Thresh, cur_Label); 
     } 
     //Down Point 
     if (next_Label_rowPtr[cur_SP.x] == 0 && std::abs(cur_Input_rowPtr[cur_SP.x] - next_Input_rowPtr[cur_SP.x]) < Thresh){ 
      cv::Point2i down_Point(cur_SP.x, cur_SP.y + 1); 

      next_Label_rowPtr[cur_SP.x] = cur_Label; 
      DFS_Traversal(InputMat, LabelMat, down_Point, Thresh, cur_Label); 
     } 

    } 
    return; 

} 

8G RAM 위에 플로우없이 최대 면적 랩톱에서 실행 재귀 5000 레벨 근처에있는 72 * 72이다. DFS로 어떻게 더 잘 할 수 있습니까?

답변

1

반복을 루프로 바꾸고 명시 적 스택을 사용합니다 (모든 목록에서 수행).

스택은 호출 스택을 시뮬 레이팅하지만 너무 엄격하게 제한되지 않습니다.

Wikipedia에서 반복 구현을 참조하십시오

iterativeInorder(node) 
    s ← empty stack 
    while (not s.isEmpty() or node ≠ null) 
    if (node ≠ null) 
     s.push(node) 
     node ← node.left 
    else 
     node ← s.pop() 
     visit(node) 
     node ← node.right