알고리즘/BAEKJOON

BOJ 1395 : 스위치

피로물든딸기 2023. 1. 19. 22:40
반응형

알고리즘 문제 전체 링크

 

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

 

참고

- 구간 합 구하기 with 탑 다운 세그먼트 트리 (Top-Down Segment Tree)
- 구간 합 구하기 2 with 나중에 업데이트하기 (Top-Down Segment Tree with Lazy Propagation)

 

 

Si ~ Ti까지 스위치를 반전시켜야 하므로, 구간의 갱신이 매우 많다.

따라서 느리게 갱신되는 세그먼트 트리를 이용해서 문제를 풀 수 있다.

 

예를 들어서 1 ~ 16까지 스위치가 있고 1, 2, 4, 5번 스위치가 켜져있다고 하자.

 

이 상태에서 1 ~ 4번 스위치를 반전시키면 아래와 같이 트리가 갱신되어야 한다.

 

1 ~ 4번의 스위치가 3개 켜져 있는 상태에서 4개의 스위치를 반전시킨다는 것은

전체 스위치의 개수 (4 - 1 + 1)개의 스위치에서 3을 뺀 것과 같은 결과다.

따라서 lazy를 이용하면 아래와 같이 표시할 수 있다.

 

1 ~ 4번 트리는 4개의 구간에서 원래의 값 3을 빼고, 아래의 2개의 노드에 갱신이 필요하다고 lazy를 체크해주면 된다.

 

그리고 이 lazy는 1일 때만 update를 한다.

lazy가 짝수라면 껐다 켜는 것이므로 원래대로 돌아갈 뿐이다.

따라서 lazy_update는 아래와 같이 구현된다.

void lazy_update(int left, int right, int node)
{
	if (lazy[node] == 0) return;

	if (lazy[node] & 1) tree[node] = (right - left + 1) - tree[node];

	if (left != right)
	{
		lazy[node * 2] += lazy[node];
		lazy[node * 2 + 1] += lazy[node];
	}

	lazy[node] = 0;
}

 

전체 코드는 다음과 같다.

#include <stdio.h>

int N, M;

int A[100100];
int tree[400400];
int lazy[400400];

void lazy_update(int left, int right, int node)
{
	if (lazy[node] == 0) return;

	if (lazy[node] & 1) tree[node] = (right - left + 1) - tree[node];

	if (left != right)
	{
		lazy[node * 2] += lazy[node];
		lazy[node * 2 + 1] += lazy[node];
	}

	lazy[node] = 0;
}

void update(int left, int right, int node, int a, int b, int sw)
{
	lazy_update(left, right, node);

	if (b < left || right < a) return;

	if (a <= left && right <= b)
	{
		tree[node] = (right - left + 1) - tree[node];

		if (left != right)
		{
			lazy[node * 2] += sw;
			lazy[node * 2 + 1] += sw;
		}

		return;
	}

	int mid = (left + right) / 2;

	update(left, mid, node * 2, a, b, sw);
	update(mid + 1, right, node * 2 + 1, a, b, sw);

	tree[node] = tree[node * 2] + tree[node * 2 + 1];
}

int getSum(int left, int right, int a, int b, int node)
{
	lazy_update(left, right, node);

	if (b < left || right < a) return 0;
	if (a <= left && right <= b) return tree[node];

	int mid = (left + right) / 2;

	return getSum(left, mid, a, b, node * 2) + getSum(mid + 1, right, a, b, node * 2 + 1);
}

int main(void)
{
	scanf("%d %d", &N, &M);

	for (int i = 0; i < M; i++)
	{
		int o, s, t;

		scanf("%d %d %d", &o, &s, &t);

		if (o) printf("%d\n", getSum(1, N, s, t, 1));
		else update(1, N, 1, s, t, 1);
	}

	return 0;
}
반응형