C 언어로 깊이 제한 탐색(DFS) 구현하기

깊이 제한 탐색(Depth-Limited Search, DLS)은 그래프나 트리 데이터 구조를 탐색할 때 사용되는 깊이 우선 탐색(DFS)의 변형입니다. 특정 깊이까지의 탐색만 허용하여 무한 반복을 방지하거나, 탐색 효율성을 높이는 데 유용합니다. 특히 상태 공간이 매우 크거나, 경로 탐색에서 최적의 해를 찾을 필요가 없는 경우에 효과적입니다. 본 기사에서는 C 언어로 깊이 제한 탐색을 구현하는 방법과 응용 사례를 단계별로 소개합니다.

목차

깊이 제한 탐색(DFS)의 정의와 특징


깊이 제한 탐색(Depth-Limited Search, DLS)은 깊이 우선 탐색(DFS)의 확장으로, 특정 깊이 이상을 탐색하지 않도록 제한을 설정한 알고리즘입니다. 이는 무한 루프를 방지하거나 탐색 공간을 효과적으로 제어하기 위해 사용됩니다.

DFS와 깊이 제한 DFS의 관계


DFS는 그래프나 트리의 모든 노드를 탐색하는 알고리즘으로, 한 경로를 끝까지 탐색한 뒤 백트래킹을 통해 다른 경로를 탐색합니다. 깊이 제한 DFS는 이러한 DFS에 탐색 깊이의 상한선을 두어, 지정된 깊이를 초과하지 않도록 설계되었습니다.

깊이 제한 DFS의 특징

  1. 효율성: 제한된 깊이만 탐색하기 때문에 대규모 그래프에서도 실행 시간이 짧아질 수 있습니다.
  2. 적용성: 무한 그래프나 깊이를 제한할 필요가 있는 문제(예: 최적 경로 탐색)에서 유용합니다.
  3. 단점: 제한된 깊이 때문에 최적 해를 찾지 못할 가능성이 있습니다.

활용 예시

  • 퍼즐 문제(예: 8 퍼즐, 체스 엔드게임)에서의 탐색
  • 네트워크 패킷 라우팅 경로 제한
  • 상태 공간이 크고 깊이가 제한된 경우의 문제 해결

깊이 제한 탐색은 깊이를 효율적으로 제어하며 다양한 문제를 해결할 수 있는 강력한 알고리즘입니다.

깊이 제한 DFS 알고리즘의 동작 원리

깊이 제한 DFS는 일반 DFS의 탐색 원리를 기반으로 하며, 탐색 깊이를 제한하는 추가 조건을 적용해 작동합니다. 이 알고리즘은 그래프나 트리를 탐색하면서, 지정된 깊이에 도달하면 더 이상 하위 노드로 탐색하지 않고 백트래킹을 수행합니다.

알고리즘 동작 단계

  1. 초기화: 그래프 구조와 시작 노드, 그리고 탐색 제한 깊이를 설정합니다.
  2. 탐색 조건 확인: 현재 노드의 깊이가 제한 깊이를 초과하는지 확인합니다.
  3. 탐색 진행: 제한 깊이를 초과하지 않으면, 현재 노드를 방문하고 인접 노드로 탐색을 진행합니다.
  4. 백트래킹: 인접 노드가 없거나 제한 깊이를 초과하면 백트래킹하여 이전 노드로 돌아갑니다.
  5. 탐색 종료: 그래프의 모든 노드가 탐색되거나 제한 조건에 따라 탐색이 중단되면 종료합니다.

재귀적 구현의 의사 코드

void depthLimitedDFS(Node* currentNode, int currentDepth, int maxDepth) {
    if (currentDepth > maxDepth) {
        return; // 제한 깊이를 초과하면 탐색 종료
    }

    visit(currentNode); // 현재 노드 방문

    for (Node* neighbor : currentNode->neighbors) {
        depthLimitedDFS(neighbor, currentDepth + 1, maxDepth); // 인접 노드 탐색
    }
}

핵심 로직 설명

  • 현재 깊이 비교: 현재 깊이가 maxDepth를 초과하면 탐색을 종료하여 무한 루프를 방지합니다.
  • 재귀 호출: 인접 노드에 대해 재귀적으로 탐색을 수행하며, 깊이를 1씩 증가시킵니다.
  • 백트래킹: 더 이상 탐색할 노드가 없거나 제한에 도달하면 이전 호출로 돌아갑니다.

예시 입력과 출력

입력 그래프:

A -> B, C  
B -> D  
C -> E, F  

제한 깊이: 2

출력:

방문 순서: A, B, C, D, E, F (깊이 제한 적용)

깊이 제한 DFS는 탐색 범위를 효과적으로 제한하면서도 그래프 탐색의 전반적인 특성을 유지합니다. 이를 통해 특정 깊이에서 필요한 데이터만 빠르게 검색할 수 있습니다.

C 언어에서 DFS 구현의 기본 구조

C 언어에서 깊이 제한 탐색(DFS)을 구현하기 위해 필요한 기본적인 데이터 구조와 함수 구성을 이해해야 합니다. 이를 통해 효율적으로 그래프를 탐색하고 제한 조건을 적용할 수 있습니다.

필수 데이터 구조

  1. 그래프 표현
    그래프는 일반적으로 인접 리스트 또는 인접 행렬로 표현합니다. 깊이 제한 DFS에서는 인접 리스트를 사용하는 것이 메모리와 탐색 효율 측면에서 유리합니다.
typedef struct Node {
    int vertex;
    struct Node* next;
} Node;

typedef struct Graph {
    int numVertices;
    Node** adjLists;
    int* visited;
} Graph;
  1. 스택 구조
    DFS는 스택을 활용하는 탐색 방식입니다. 재귀 함수 호출로 내부 스택을 사용할 수 있지만, 명시적인 스택을 구현할 수도 있습니다.

기본 함수 구성

  1. 그래프 초기화
    그래프를 동적으로 생성하고, 노드와 간선을 설정합니다.
Graph* createGraph(int vertices) {
    Graph* graph = malloc(sizeof(Graph));
    graph->numVertices = vertices;
    graph->adjLists = malloc(vertices * sizeof(Node*));
    graph->visited = malloc(vertices * sizeof(int));

    for (int i = 0; i < vertices; i++) {
        graph->adjLists[i] = NULL;
        graph->visited[i] = 0;
    }
    return graph;
}
  1. 노드 연결
    그래프에 간선을 추가합니다.
void addEdge(Graph* graph, int src, int dest) {
    Node* newNode = malloc(sizeof(Node));
    newNode->vertex = dest;
    newNode->next = graph->adjLists[src];
    graph->adjLists[src] = newNode;
}
  1. DFS 탐색 함수
    DFS 탐색과 깊이 제한을 적용하는 함수를 정의합니다.
void depthLimitedDFS(Graph* graph, int vertex, int depth, int maxDepth) {
    if (depth > maxDepth) {
        return; // 깊이 제한 초과 시 종료
    }

    graph->visited[vertex] = 1;
    printf("Visited %d\n", vertex);

    Node* temp = graph->adjLists[vertex];
    while (temp) {
        int connectedVertex = temp->vertex;
        if (graph->visited[connectedVertex] == 0) {
            depthLimitedDFS(graph, connectedVertex, depth + 1, maxDepth);
        }
        temp = temp->next;
    }
}

작동 원리

  1. 그래프 생성 및 간선 추가: createGraphaddEdge를 사용해 탐색할 그래프를 초기화합니다.
  2. DFS 호출: 탐색을 시작할 노드와 깊이 제한을 인수로 하여 depthLimitedDFS를 호출합니다.
  3. 방문 기록: 각 노드를 방문할 때 visited 배열을 업데이트하여 무한 루프를 방지합니다.
  4. 깊이 제한 적용: 재귀 호출 시 현재 깊이가 제한 깊이를 초과하면 탐색을 중단합니다.

다음 단계


이 기본 구조를 기반으로 실제로 동작하는 깊이 제한 DFS 구현 코드를 작성하고, 이를 실행하여 동작을 확인할 수 있습니다.

깊이 제한을 추가한 DFS 구현 코드

아래는 C 언어로 구현한 깊이 제한 탐색(DFS) 코드입니다. 그래프는 인접 리스트로 표현되며, 재귀 호출을 통해 깊이를 제한하여 탐색을 수행합니다.

전체 코드

#include <stdio.h>
#include <stdlib.h>

// 노드 구조체 정의
typedef struct Node {
    int vertex;
    struct Node* next;
} Node;

// 그래프 구조체 정의
typedef struct Graph {
    int numVertices;
    Node** adjLists;
    int* visited;
} Graph;

// 그래프 생성
Graph* createGraph(int vertices) {
    Graph* graph = malloc(sizeof(Graph));
    graph->numVertices = vertices;
    graph->adjLists = malloc(vertices * sizeof(Node*));
    graph->visited = malloc(vertices * sizeof(int));

    for (int i = 0; i < vertices; i++) {
        graph->adjLists[i] = NULL;
        graph->visited[i] = 0;
    }
    return graph;
}

// 간선 추가
void addEdge(Graph* graph, int src, int dest) {
    Node* newNode = malloc(sizeof(Node));
    newNode->vertex = dest;
    newNode->next = graph->adjLists[src];
    graph->adjLists[src] = newNode;
}

// 깊이 제한 DFS 구현
void depthLimitedDFS(Graph* graph, int vertex, int depth, int maxDepth) {
    if (depth > maxDepth) {
        return; // 깊이 제한 초과 시 탐색 종료
    }

    graph->visited[vertex] = 1;
    printf("Visited %d at depth %d\n", vertex, depth);

    Node* temp = graph->adjLists[vertex];
    while (temp) {
        int connectedVertex = temp->vertex;
        if (graph->visited[connectedVertex] == 0) {
            depthLimitedDFS(graph, connectedVertex, depth + 1, maxDepth);
        }
        temp = temp->next;
    }
}

// 메인 함수
int main() {
    int vertices = 6; // 노드 개수
    Graph* graph = createGraph(vertices);

    addEdge(graph, 0, 1);
    addEdge(graph, 0, 2);
    addEdge(graph, 1, 3);
    addEdge(graph, 1, 4);
    addEdge(graph, 2, 5);

    int maxDepth = 2; // 탐색 깊이 제한
    printf("Depth-Limited DFS starting from vertex 0 with max depth %d:\n", maxDepth);
    depthLimitedDFS(graph, 0, 0, maxDepth);

    // 메모리 해제 (간단히 처리)
    free(graph->visited);
    free(graph->adjLists);
    free(graph);

    return 0;
}

코드 해설

  1. 그래프 생성 및 간선 추가
  • createGraph 함수는 노드와 간선을 동적으로 할당하여 그래프를 초기화합니다.
  • addEdge 함수는 특정 노드 간의 연결을 설정합니다.
  1. 깊이 제한 DFS 구현
  • depthLimitedDFS 함수는 현재 깊이를 확인하고, 제한 깊이를 초과하면 탐색을 종료합니다.
  • 방문한 노드는 visited 배열에 기록되어 다시 방문하지 않도록 합니다.
  1. 메인 함수
  • 그래프를 초기화하고 간선을 추가한 뒤, 시작 노드와 탐색 제한 깊이를 설정하여 DFS를 호출합니다.

출력 예시

그래프:

0 -> 1, 2  
1 -> 3, 4  
2 -> 5  

탐색 깊이 제한: 2

출력:

Depth-Limited DFS starting from vertex 0 with max depth 2:  
Visited 0 at depth 0  
Visited 1 at depth 1  
Visited 2 at depth 1  
Visited 3 at depth 2  
Visited 4 at depth 2  
Visited 5 at depth 2  

적용 사례


이 구현 코드는 작은 규모의 그래프 탐색이나 특정 깊이에서의 탐색 문제를 해결하는 데 사용할 수 있습니다. 필요에 따라 그래프 구조를 확장하거나 탐색 조건을 변경하여 활용할 수 있습니다.

깊이 제한 DFS의 응용 예시

깊이 제한 탐색(DFS)은 특정 조건과 제약 아래에서 탐색 문제를 해결하는 데 유용합니다. 아래는 다양한 실제 응용 사례를 소개합니다.

1. 상태 공간 탐색


깊이 제한 DFS는 퍼즐 문제나 상태 공간 탐색에서 활용될 수 있습니다.
예를 들어, 8 퍼즐 문제에서 특정 이동 횟수 이하로 해결 가능한 상태를 탐색하려면 깊이 제한을 사용하여 불필요한 계산을 줄일 수 있습니다.

응용 예시:

  • 시작 상태:
  1 2 3  
  4 5 6  
  7 8 _  
  • 목표 상태:
  1 2 3  
  4 5 6  
  7 _ 8  
  • 최대 이동 횟수 제한: 3
    이 경우, 탐색 깊이를 3으로 제한하여 문제를 해결할 수 있습니다.

2. 경로 탐색 문제


그래프에서 깊이 제한 DFS는 특정 조건에 따라 경로를 탐색하는 문제를 해결하는 데 유용합니다.
예를 들어, 네트워크 라우팅에서 특정 홉 수 이내에 도달할 수 있는 경로를 찾는 데 사용됩니다.

응용 예시:

  • 그래프:
  A -> B -> C -> D  
  A -> E -> F -> D  
  • 제한 깊이: 2
  • 결과: 깊이 제한 DFS를 사용하면 A -> B -> C 경로만 탐색됩니다.

3. 게임 트리 탐색


체스와 같은 게임에서 특정 깊이만 탐색하여 가능한 이동을 평가하는 데 깊이 제한 DFS가 사용됩니다. 이는 미니맥스 알고리즘이나 알파-베타 가지치기와 같은 알고리즘의 기반이 됩니다.

응용 예시:

  • 현재 보드 상태에서 최대 3턴 후의 결과를 예측.
  • 깊이 제한 DFS를 사용해 3턴 이후의 상태는 무시.

4. 네트워크 분석


깊이 제한 DFS는 네트워크 그래프에서 특정 거리 내의 노드를 탐색하거나, 특정 범위 내의 연결 관계를 분석하는 데 사용됩니다.
예를 들어, 소셜 네트워크 분석에서 특정 사용자와 N단계 이내로 연결된 사용자들을 찾는 데 활용할 수 있습니다.

5. 제한된 자원의 탐색


로봇 탐사와 같은 물리적 탐사에서는 배터리나 시간과 같은 자원 제한이 있습니다. 깊이 제한 DFS는 탐사 범위를 제어하여 효율적으로 탐사할 수 있도록 합니다.

응용 예시:

  • 로봇이 지도 탐색 시 배터리 잔량에 따라 깊이 제한을 설정.

장점과 단점

  • 장점:
  • 탐색 범위를 제한하여 시간과 메모리 사용량을 줄임.
  • 무한 그래프에서도 안정적으로 동작.
  • 단점:
  • 깊이 제한이 너무 작을 경우 최적 해를 놓칠 가능성이 있음.
  • 깊이 제한을 잘못 설정하면 성능에 영향을 줄 수 있음.

응용을 위한 팁

  • 탐색 문제에서 최적 깊이 제한을 찾기 위해 문제의 특성과 입력 데이터의 특성을 분석합니다.
  • 깊이 제한 DFS를 다른 알고리즘(예: IDDFS, A*)과 결합하여 활용도를 높일 수 있습니다.

깊이 제한 DFS는 다양한 실생활 문제를 효율적으로 해결할 수 있는 강력한 도구입니다. 이를 활용해 복잡한 탐색 문제를 간단히 해결할 수 있습니다.

일반 DFS와 깊이 제한 DFS의 차이점

깊이 제한 탐색(DFS)은 일반 DFS에 탐색 깊이의 상한선을 추가한 알고리즘입니다. 두 탐색 방식은 기본적으로 그래프나 트리를 깊이 우선으로 탐색하지만, 적용 목적과 탐색 범위에서 차이가 있습니다.

1. 탐색 범위

  • 일반 DFS: 그래프의 모든 노드를 탐색하며, 사이클이 있는 경우 무한 루프에 빠질 위험이 있습니다.
  • 깊이 제한 DFS: 탐색 깊이에 제한을 두어, 지정된 깊이를 초과하는 노드는 탐색하지 않습니다.

비교 예시:
그래프 구조:

A -> B -> C -> D  
A -> E -> F -> G  
  • 일반 DFS:
  A -> B -> C -> D  
  A -> E -> F -> G  

그래프 전체를 탐색.

  • 깊이 제한 DFS (깊이 제한: 2):
  A -> B -> C  
  A -> E -> F  

2단계 이후 노드는 탐색하지 않음.

2. 무한 그래프 처리

  • 일반 DFS: 무한 그래프에서는 무한 루프에 빠질 가능성이 큽니다.
  • 깊이 제한 DFS: 탐색 깊이에 제한을 둬 무한 루프를 방지합니다.

적용 예시:

  • 네트워크 경로 탐색에서 깊이 제한을 통해 특정 범위 내에서만 탐색을 수행.

3. 시간 및 공간 복잡도

  • 일반 DFS: 시간 복잡도는 O(V + E), 공간 복잡도는 O(V)입니다. (V는 노드 수, E는 간선 수)
  • 깊이 제한 DFS: 시간 및 공간 복잡도는 제한된 깊이에 따라 감소합니다.

비교 예시:

  • 깊이 제한 DFS의 복잡도는 O(b^d) (b: 노드의 평균 차수, d: 제한 깊이)로, 그래프 전체를 탐색하는 경우보다 효율적입니다.

4. 알고리즘의 목적

  • 일반 DFS: 그래프의 모든 노드와 경로를 탐색하며, 완전한 탐색을 수행합니다.
  • 깊이 제한 DFS: 특정 깊이까지만 탐색하여, 제한된 조건 내에서 효율적인 탐색을 목표로 합니다.

5. 주요 활용 사례

일반 DFS깊이 제한 DFS
모든 경로 탐색특정 깊이까지 경로 탐색
사이클 검출무한 루프 방지
그래프의 연결 요소 확인제한된 상태 공간 탐색
위상 정렬제한된 네트워크 분석

6. 구현 차이

일반 DFS:

void DFS(Graph* graph, int vertex) {
    graph->visited[vertex] = 1;
    printf("Visited %d\n", vertex);

    Node* temp = graph->adjLists[vertex];
    while (temp) {
        int connectedVertex = temp->vertex;
        if (graph->visited[connectedVertex] == 0) {
            DFS(graph, connectedVertex);
        }
        temp = temp->next;
    }
}

깊이 제한 DFS:

void depthLimitedDFS(Graph* graph, int vertex, int depth, int maxDepth) {
    if (depth > maxDepth) {
        return; // 깊이 제한 초과 시 종료
    }

    graph->visited[vertex] = 1;
    printf("Visited %d at depth %d\n", vertex, depth);

    Node* temp = graph->adjLists[vertex];
    while (temp) {
        int connectedVertex = temp->vertex;
        if (graph->visited[connectedVertex] == 0) {
            depthLimitedDFS(graph, connectedVertex, depth + 1, maxDepth);
        }
        temp = temp->next;
    }
}

요약


일반 DFS는 전체 그래프 탐색에 적합하며, 깊이 제한 DFS는 특정 조건이나 자원이 제한된 환경에서 효율적으로 활용됩니다. 문제의 목적에 따라 적절한 알고리즘을 선택하는 것이 중요합니다.

구현 시 자주 발생하는 오류와 디버깅 팁

C 언어로 깊이 제한 탐색(DFS)을 구현할 때는 데이터 구조의 설정, 재귀 호출의 한계, 깊이 제한 처리와 관련된 여러 오류가 발생할 수 있습니다. 이러한 오류를 효과적으로 해결하기 위한 방법과 디버깅 팁을 소개합니다.

1. 인접 리스트 초기화 오류


문제: 그래프의 인접 리스트를 초기화하지 않거나, 간선을 잘못 추가하는 경우 올바르게 탐색되지 않을 수 있습니다.

해결 방안:

  • createGraph 함수에서 각 노드의 초기값을 NULL로 설정했는지 확인합니다.
  • addEdge 함수에서 메모리를 동적으로 할당하고, 각 간선이 제대로 연결되었는지 검증합니다.

디버깅 팁:

  • 그래프 구조를 출력하여 노드 간 연결 상태를 확인합니다.
void printGraph(Graph* graph) {
    for (int i = 0; i < graph->numVertices; i++) {
        Node* temp = graph->adjLists[i];
        printf("\n Vertex %d:", i);
        while (temp) {
            printf(" -> %d", temp->vertex);
            temp = temp->next;
        }
    }
}

2. 방문 배열 초기화 누락


문제: visited 배열이 초기화되지 않으면 이전 탐색 결과가 남아 있어 무한 루프나 중복 방문 문제가 발생할 수 있습니다.

해결 방안:

  • 탐색을 시작하기 전에 visited 배열을 반드시 0으로 초기화합니다.
for (int i = 0; i < graph->numVertices; i++) {
    graph->visited[i] = 0;
}

3. 깊이 제한 조건 누락


문제: 재귀 호출에서 깊이 제한 조건을 확인하지 않으면 지정된 깊이를 초과한 노드도 탐색될 수 있습니다.

해결 방안:

  • depthLimitedDFS 함수에서 depth > maxDepth 조건을 먼저 확인하고 탐색을 종료합니다.
if (depth > maxDepth) {
    return; // 제한 깊이를 초과하면 종료
}

4. 메모리 누수


문제: 동적으로 할당된 메모리를 해제하지 않으면 메모리 누수가 발생합니다.

해결 방안:

  • 탐색 종료 후 모든 동적으로 할당된 노드와 배열을 free 함수로 해제합니다.
void freeGraph(Graph* graph) {
    for (int i = 0; i < graph->numVertices; i++) {
        Node* temp = graph->adjLists[i];
        while (temp) {
            Node* toFree = temp;
            temp = temp->next;
            free(toFree);
        }
    }
    free(graph->adjLists);
    free(graph->visited);
    free(graph);
}

5. 스택 오버플로우


문제: 탐색 깊이가 너무 크거나 그래프가 매우 크면 재귀 호출이 과도하게 이루어져 스택 오버플로우가 발생합니다.

해결 방안:

  • 깊이 제한을 적절히 설정하여 탐색 범위를 조정합니다.
  • 명시적 스택을 사용하여 비재귀적으로 구현합니다.
void iterativeDepthLimitedDFS(Graph* graph, int startVertex, int maxDepth) {
    Stack* stack = createStack();
    push(stack, startVertex, 0);

    while (!isEmpty(stack)) {
        StackNode* current = pop(stack);
        int vertex = current->vertex;
        int depth = current->depth;

        if (depth > maxDepth) continue;

        if (graph->visited[vertex] == 0) {
            graph->visited[vertex] = 1;
            printf("Visited %d at depth %d\n", vertex, depth);
        }

        Node* temp = graph->adjLists[vertex];
        while (temp) {
            if (graph->visited[temp->vertex] == 0) {
                push(stack, temp->vertex, depth + 1);
            }
            temp = temp->next;
        }
    }
}

6. 디버깅 및 테스트 전략

  • 소규모 그래프 테스트: 노드 수와 간선 수가 적은 그래프를 사용하여 기본적인 탐색이 올바르게 수행되는지 확인합니다.
  • 경계값 테스트: 깊이 제한이 0인 경우, 최대 깊이와 같은 경우 등 경계값에서의 동작을 확인합니다.
  • 로그 출력: 탐색 중 각 노드와 깊이를 출력하여 동작을 추적합니다.

요약


깊이 제한 DFS를 구현할 때는 데이터 구조 초기화, 깊이 제한 조건 확인, 메모리 관리 등을 신중히 처리해야 합니다. 위의 디버깅 팁을 활용하면 오류를 효과적으로 식별하고 해결할 수 있습니다.

요약

깊이 제한 탐색(DFS)은 그래프나 트리 탐색에서 특정 깊이까지 탐색을 제한하여 효율성을 높이고 무한 루프를 방지하는 알고리즘입니다. 본 기사에서는 깊이 제한 DFS의 개념, C 언어에서의 구현 방법, 응용 사례, 그리고 구현 시 자주 발생하는 문제와 해결 방안을 다루었습니다.

깊이 제한 DFS는 대규모 그래프 탐색, 퍼즐 문제 해결, 네트워크 분석 등 다양한 분야에서 유용하게 활용될 수 있습니다. 이를 올바르게 구현하고 디버깅하면 복잡한 탐색 문제를 효과적으로 해결할 수 있습니다.

목차