[ABC223F] Parenthesis Checking 题解

[ABC223F] Parenthesis Checking 题解

思路解析

在开始之前,首先我们需要知道合法括号序列的判断方法。我们可以给每个括号打上权值,设左括号权值为 \(1\),右括号权值为 \(-1\),这样一个 \(\texttt{(()())}\) 括号串用数字存下就是 \(1,1,-1,1,-1,-1\),这时我们再给序列计算一下前缀和就成了 \(1,2,1,2,1,0\)。此时我们发现序列有一个性质就是元素全部大于等于 \(0\),同时结尾的元素一定为 \(0\)。而例如我们找一个括号序列 \(\texttt{)()(}\),它的前缀和数组为 \(-1,0,-1,0\),可见虽然结尾是 \(0\),但中间有元素小于 \(0\),因此该括号序列并不合法。

现在我们已经知道了如何通过序列的前缀和数组判断该序列是否合法,接下来我们就考虑如何修改。我们可以先把前缀和数组存下来,然后考虑每一次交换会对这个数组产生怎样的影响。首先,若交换 \(l,r\) 两个位置上的括号,我们可以发现区间 \([1,l-1]\)\([r,n]\) 是没有影响的,真正有影响的只有 \([l,r-1]\) 这个区间。接下来我们分两种情况考虑:

  • \(l\) 位上的括号交换前是左括号时,可见由于第 \(l\) 位从 \(1\) 改成了 \(-1\),所以就会对该区间加上 \(-2\) 的权值。
  • \(l\) 位上的括号交换前是右括号时,可见由于第 \(l\) 位从 \(-1\) 改成了 \(1\),所以就会对该区间加上 \(2\) 的权值。

最后考虑如何实现,我们需要做到区间求最小值,单点查询,区间加三种操作,可以想到使用线段树维护。注意由于我们存的值是前缀和,所以需要减去 \(l-1\) 位置上的值才能得到正确的值。记得特判 \(l-1=0\) 时的情况。

code

#include<bits/stdc++.h>
using namespace std;
const int N = 2e5 + 10, M = 8e5 + 10;
string str;
int n, q, a[N], s[M], t[M];
void push_up(int p) {
	int ls = (p << 1), rs = (p << 1) + 1;
	s[p] = min(s[ls], s[rs]);
}
void build(int p, int l, int r) {
	if(l == r) {
		s[p] = a[l];
		return;
	}
	int m = l + ((r - l) >> 1), ls = (p << 1), rs = (p << 1) + 1;
	build(ls, l, m);
	build(rs, m + 1, r);
	push_up(p);
}
void addt(int p, int l, int r, int k) {
	s[p] += k;
	t[p] += k;
}
void push_down(int p, int l, int r) {
	if(!t[p]) return;
	int m = l + ((r - l) >> 1), ls = (p << 1), rs = (p << 1) + 1;
	addt(ls, l, m, t[p]);
	addt(rs, m + 1, r, t[p]);
	t[p] = 0;
}
void add(int p, int l, int r, int x, int y, int k) {
	if(r < x || l > y) return;
	if(l >= x && r <= y) {
		addt(p, l, r, k);
		return;
	}
	push_down(p, l, r);
	int m = l + ((r - l) >> 1), ls = (p << 1), rs = (p << 1) + 1;
	add(ls, l, m, x, y, k);
	add(rs, m + 1, r, x, y, k);
	push_up(p);
}
int ask(int p, int l, int r, int x, int y) {
	if(r < x || l > y) return 2e9;
	if(l >= x && r <= y) return s[p];
	push_down(p, l, r);
	int m = l + ((r - l) >> 1), ls = (p << 1), rs = (p << 1) + 1;
	return min(ask(ls, l, m, x, y), ask(rs, m + 1, r, x, y));
}
int main() {
	cin >> n >> q;
	cin >> str;
	str = ' ' + str;
	for(int i = 1; i <= n; i++) {
		a[i] = a[i - 1];
		if(str[i] == '(') a[i]++;
		else a[i]--;
	}
	build(1, 1, n);
	while(q--) {
		int op, l, r;
		cin >> op >> l >> r;
		if(op == 1) {
			if(str[l] != str[r]) {
				if(str[l] == '(') add(1, 1, n, l, r - 1, -2);
				else add(1, 1, n, l, r - 1, 2);
				swap(str[l], str[r]);
			}
		}
		else {
			int tmp = ask(1, 1, n, l - 1, l - 1);
			if(l == 1) tmp = 0;
			int t1 = ask(1, 1, n, l, r - 1) - tmp, t2 = ask(1, 1, n, r, r) - tmp;
			if(t1 >= 0 && t2 == 0) cout << "Yes\n";
			else cout << "No\n";
		}
	}
	return 0;
}
posted @ 2024-04-04 13:16  2020luke  阅读(30)  评论(0)    收藏  举报