와이유스토리

[트리(LCA)] 백준 11437 LCA C++ 본문

코딩테스트/그래프|트리

[트리(LCA)] 백준 11437 LCA C++

유(YOO) 2022. 1. 26. 21:11

 

https://www.acmicpc.net/problem/11437

 

11437번: LCA

첫째 줄에 노드의 개수 N이 주어지고, 다음 N-1개 줄에는 트리 상에서 연결된 두 정점이 주어진다. 그 다음 줄에는 가장 가까운 공통 조상을 알고싶은 쌍의 개수 M이 주어지고, 다음 M개 줄에는 정

www.acmicpc.net

 

#include <iostream>
#include <vector>
#define size 50001

using namespace std;

vector<int> graph[size];

int level[size];
int dp[size][30]; // log2(50001)
int visited[size];
int n;

void dfs(int v, int depth) {
	visited[v] = 1;
	level[v] = depth;

	for (auto i : graph[v]) {
		if (!visited[i]) {
			dp[i][0] = v; // dp[자식 노드][0] = 부모 노드
			dfs(i, depth + 1);
		}
	}
}

void initTable() { // i의 2^j번째 조상 = (i의 2^(j-1)번째 조상)의 2^(j-1)번째 조상
	for (int j = 1; j < 30; j++) {
		for (int i = 1; i <= n; i++) {
			dp[i][j] = dp[dp[i][j - 1]][j - 1]; // dp[i][j] := i의 2^j번째 조상
		}
	}
}

int lca(int u, int v) {
	if (level[u] < level[v]) swap(u, v);
	int diff = level[u] - level[v];
	int i = 0;
	while(diff) { // 레벨 차이
		if (diff & 1) u = dp[u][i];
		diff >>= 1;
		i++;
	}
	if (u == v) return u; // 같은 레벨

	for (int i = 29; i >= 0; i--) { // 조상이 다르면 노드들을 조상 노드로 변경하여 다시 찾음
		if (dp[u][i] != dp[v][i]) u = dp[u][i], v = dp[v][i];
	}

	return dp[u][0]; // u!=v (u와 v의 부모 노드 같음)
}

int main() {
	ios_base::sync_with_stdio(0); 
	cin.tie(NULL); cout.tie(NULL);

	int m, a, b;

	cin >> n;
	for (int i = 0; i < n - 1; i++) {
		cin >> a >> b;
		graph[a].push_back(b);
		graph[b].push_back(a);
	}

	dfs(1, 0);
	initTable();

	cin >> m;
	for(int i = 0; i < m; i++) {
		cin >> a >> b;
		cout << lca(a, b) << "\n";
	}
}
Comments