首页 > 其他分享 >[ABC341E] Alternating String 题解

[ABC341E] Alternating String 题解

时间:2024-02-27 19:55:36浏览次数:50  
标签:ABC341E rt ch int 题解 tre mid Alternating query

题目传送门

原题传送门

题意

给出长为 \(n\) 的 01 串,如果一个子串 01 交替出现,则称其为“好的”。有 \(q\) 次询问,把 \([x,y]\) 中的每一位反转或者询问 \([x,y]\) 是否是“好的”。

分析

一眼线段树。

用线段树维护区间是否是“好的”,每个节点维护最左段和最右端的值,pushupquery 向上合并区间时判断即可。

下面的代码有比较详细的注释,具体看代码即可,看不懂可以画图辅助理解。

代码

#include <bits/stdc++.h>
#define int long long
#define N 500005
using namespace std;
int n, q, a[N];

inline int read(int &x) {
	char ch = x = 0;
	int m = 1;
	while (ch < '0' || ch > '9') {
		ch = getchar();
		if (ch == '-') m *= -1;
	}
	while (ch >= '0' && ch <= '9') {
		x = (x << 1) + (x << 3) + ch - 48;
		ch = getchar();
	}
	x *= m;
	return x;
}

struct node {
	int l, r, lval, rval, tag, isg;
};

struct segtre {
	node tre[N << 1]; //动态开点节省空间
	int root, tot;
	inline void pushup(int rt) { //合并两个区间
		int l = tre[rt].l, r = tre[rt].r;
		tre[rt].isg = (tre[l].rval ^ tre[r].lval) & tre[l].isg & tre[r].isg; //左右两个区间都是好的并且左区间右侧和右区间左侧不同时当前的区间才是好的
		tre[rt].lval = tre[l].lval;
		tre[rt].rval = tre[r].rval;
		return ;
	}
	inline void pushdown(int rt) { //下传反转标记
		if (!tre[rt].tag) return ;
		int l = tre[rt].l, r = tre[rt].r;
		tre[l].tag ^= 1, tre[r].tag ^= 1;
		tre[l].lval ^= 1, tre[l].rval ^= 1;
		tre[r].lval ^= 1, tre[r].rval ^= 1; //全都反转
		tre[rt].tag = 0;
		return ;
	}
	void build(int &rt, int l, int r) { //建树
		if (!rt) rt = ++tot;
		if (l == r) {
			tre[rt].lval = tre[rt].rval = a[l]; //底层初值
			tre[rt].isg = 1; //长度为 1 的区间应该是好的
			return ;
		}
		int mid = (l + r) >> 1;
		build(tre[rt].l, l, mid);
		build(tre[rt].r, mid + 1, r);
		pushup(rt);
		return ;
	}
	void update(int rt, int l, int r, int L, int R) {
		if (L <= l && r <= R) { //包含就打上 tag 返回
			tre[rt].tag ^= 1;
			tre[rt].lval ^= 1;
			tre[rt].rval ^= 1;
			return ;
		}
		pushdown(rt);
		int mid = (l + r) >> 1;
		if (L <= mid) update(tre[rt].l, l, mid, L, R);
		if (R > mid) update(tre[rt].r, mid + 1, r, L, R);
		pushup(rt);
		return ;
	}
	int query(int rt, int l, int r, int L, int R) {
		if (L <= l && r <= R) {
			return tre[rt].isg;
		}
		pushdown(rt);
		int mid = (l + r) >> 1, lr = tre[rt].l, rr = tre[rt].r;
		if (L <= mid && R > mid) { //如果跨越 mid 就要合并左右两个询问区间,合并可以参考 pushup 函数里的注释
			if ((tre[lr].rval ^ tre[rr].lval) == 0) return 0;
			return query(lr, l, mid, L, R) && query(rr, mid + 1, r, L, R);
		} else if (L <= mid) { //没有跨越就直接返回就好了
			return query(lr, l, mid, L, R);
		} else {
			return query(rr, mid + 1, r, L, R);
		}
	}
} tre;

signed main() {
	read(n), read(q);
	char ch = 0;
	while (ch != '0' && ch != '1') ch = getchar();
	for (int i = 1; i <= n; i++, ch = getchar()) {
		if (ch == '1') a[i] = 1;
	}
	tre.build(tre.root, 1, n);
	int opt, l, r;
	while (q--) {
		read(opt), read(l), read(r);
		if (opt == 1) {
			tre.update(tre.root, 1, n, l, r);
		} else {
			if (tre.query(tre.root, 1, n, l, r)) {
				printf("Yes\n");
			} else {
				printf("No\n");
			}
		}
	}
	return 0;
}

标签:ABC341E,rt,ch,int,题解,tre,mid,Alternating,query
From: https://www.cnblogs.com/iloveoi/p/18037753

相关文章

  • [ABC342G] Retroactive Range Chmax 题解
    洛谷传送门原题传送门题意维护一个数列,有以下三个操作:区间最值操作,即将\([l,r]\)区间内的\(A_i\)变成\(\max(A_i,v)\)。删除操作操作,即将第\(i\)次操作删除,保证第\(i\)次操作是操作\(1\),且未被删除。注:仅删除第\(i\)次操作,后续操作仍然在。查询,询问当前的......
  • P1013 进制位 题解
    题注题目传送门这篇题解其实是上一篇题解(Llf0703同志)证明过程的完善(其实就是思路一样了啦),来让入门者或追求严谨者对证明过程更加了解。题目分析\(3\leqn\leq9\),也即数字的个数\(N\leq8\)。研究样例发现,\(N\)与进制\(R\),以及数字对应两位数个数\(M\)与数字本身......
  • Pursuit For Artifacts 题解(图论)
    PursuitForArtifacts题解题目给定一张\(n\)个点\(m\)条边的简单无向连通图,边有边权,边权要么为\(0\),要么为\(1\)。每条边只能通过一次(两个方向加起来只能通过一次)。求是否存在一条从\(a\)到\(b\)的路径,满足路径上至少存在一条权为\(1\)的边。\(1\leqn,m\l......
  • CF1864C 题解
    \(x=2^k\)是好做的,每次以\(2^{k-1}\)为因数即可。对于其他情况,考虑每次让\(x\)减去其二进制下最低位的\(1\)直至变成\(2^k\)。这种策略下显然每个数只会在以上两个大步骤下取到,故每个数使用不超过\(2\)次。同时操作次数在\(O(\logn)\)这个量级。#include<bits/......
  • P7086 题解
    考虑把每个字符串的前\(k\)位和后\(k\)位看成点,字符串看成边,那么一个字符串前缀后缀至少有一个是相似群体的前缀后缀,看成这条边的两个端点至少有一个被选中。那么这就变成了一个最小点覆盖问题。考虑匈牙利算法算出答案,然后考虑如何构造答案。考虑右边没有被匹配的点,选中这......
  • AT_abc317_f 题解
    调了一小时结果发现爆longlong了。考虑数位dp,具体来说,设计状态\(dp_{i,r_1,r_2,r_3,mx_1,mx_2,mx3_,c_1,c_2,c_3}\)表示当前考虑到第\(i\)位,\(x_1,x_2,x_3\)模\(a_1,a_2,a_3\)等于\(r_1,r_2,r_3\)三个数是否达到\(n\)的上界以及是否全部是\(0\)。然后从高到低枚......
  • AT_joisc2019_j 题解
    先考虑这个式子:\[\sum_{j=1}^{M}|C_{k_{j}}-C_{k_{j+1}}|\]一定是在\(C\)有序时取到,具体证明很简单各位读者自己证明。那么现在式子变成:\[\sum{V}+2\times({C_{\max}-C_{\min}})\]这个时候一个常见的技巧是将\(C\)排序。这个时候就可以定义状态:\[dp_{i,j}=\s......
  • CF1862G 题解
    首先这个查询操作很迷,考虑先化简查询操作。不难发现由于每次是加上一个逆的等差序列,因此一次操作完每个数与它的前驱之差一定会减少,因此加上等差序列的次数就等于全局每个数与它的前驱之差最大值。又因为会排序去重,所以最后剩下来的数一定是最开始的数一路加过来的,至此我们发现......
  • P5837 [USACO19DEC] Milk Pumping G 题解
    原题传送门思路只用堆每一个点跑一边最短路,在用当前点到点\(n\)的距离,再用当前点的\(f\)乘上\(10^6\)除以刚刚算出的值即可。代码#include<iostream>#include<cstdio>#include<cstring>#include<algorithm>#include<vector>#include<queue>usingnamespacestd;#defin......
  • P2350 [HAOI2012] 外星人 题解
    很巧妙的一道题。首先会发现如果最终\(\varphi(N)=1\)的话一定是通过很多次从\(2\)这个因子变到\(1\)的。而这个函数每迭代一次,就会有且仅有一个\(2\)的因子变为\(1\)。所以题目转化为了求\(N\)在函数迭代过程中一共会产生多少个\(2\)的因子。考虑\(\text{dp}\),设......