【算法小结】莫队算法(基础+待修 updating)

pre文章:莫队算法 --算法竞赛专题解析(26)_罗勇军的博客-CSDN博客

1.普通莫队:

配套题目:[SDOI2009] HH的项链 - 洛谷

代码过不了,数据加强了,好像只能用高级的算法,线段树 树状数组...

#include<bits/stdc++.h>
using namespace std;
const int maxn = 1e6;
struct node {
	int L, R, k;
}q[maxn];
int v[maxn], bl[maxn], ans[maxn], cnt[maxn], ANS;
bool cmp(node a, node b) {
	if (bl[a.L] != bl[b.L]) return bl[a.L] < bl[b.L];
	else {
		if (bl[a.L] & 1) return a.R > b.R;
		else return a.R < b.R;
	}
}
void del(int x) {
	cnt[v[x]]--;
	if (cnt[v[x]] == 0) ANS--;
}
void add(int x) {
	cnt[v[x]]++;
	if (cnt[v[x]] == 1) ANS++;
}
int main() {
	int n; cin >> n; int blo = sqrt(n);
	for (int i = 1; i <= n; i++) {
		cin >> v[i]; bl[i] = (i - 1) / blo + 1;
	}
	int m; cin >> m;
	for (int i = 1; i <= m; i++) {
		cin >> q[i].L >> q[i].R; q[i].k = i;
	}
	sort(q + 1, q + 1 + m, cmp);
	int L = 1, R = 0;
	for (int i = 1; i <= m; i++) {
		while (L < q[i].L) del(L++);
		while (R < q[i].R) add(++R);
		while (L > q[i].L) add(--L);
		while (R > q[i].R) del(R--);
		ans[q[i].k] = ANS;
	}
	for (int i = 1; i <= m; i++) cout << ans[i] << endl;
	return 0;
}

 2.待修莫队:

配套题目:[国家集训队] 数颜色 / 维护队列 - 洛谷

代码同上情况

#include<bits/stdc++.h>
using namespace std;
#define N 233333
#define M 1111111

int sum, cnt[M], a[N], ans[N], cntq = 0, cntr = 0, n, m, sz;

struct ques
{
	int l, r, t, id;
} qq[N], qr[N];//两个数组分解记录每一个询问以及修改的状态

inline void add(int x)
{
	sum += !cnt[x]++;
}

inline void del(int x)
{
	sum -= !--cnt[x];
}

//add与del是普通莫队原有操作

inline void upd(int x, int t)//upd是对于时间上的变化所造成变化的维护
{
	if (qq[x].l <= qr[t].l && qr[t].l <= qq[x].r)
	{
		del(a[qr[t].l]);
		add(qr[t].r);
	} //如果这个修改的值在[l,r]区间内,则其变化将对答案造成影响
	swap(a[qr[t].l], qr[t].r);//因为修改后的下一次操作一定相反(即修改该位置->还原该位置->修改该位置...如此循环),所以只需交换即可,而不需要写两个函数
}

bool cmp (const ques &a, const ques &b)
{
	return a.l / sz == b.l / sz ? a.r / sz == b.r / sz ? a.t < b.t : a.r < b.r : a.l < b.l;
}//魔改版cmp,需要判断t的大小

int main()
{
	cin >> n >> m; sz = pow(n, 0.666);//设置块的大小
	for (int i = 1; i <= n; i++) scanf("%d", a + i);
	for (int i = 1; i <= m; i++)
	{
		char op[5];
		int l, r;
		scanf("%s%d%d", op, &l, &r);
		if (op[0] == 'Q') ++cntq, qq[cntq].id = cntq, qq[cntq].l = l, qq[cntq].r = r, qq[cntq].t = cntr;//询问的时间即为该询问以前已经执行了多少次修改操作
		else qr[++cntr].l = l, qr[cntr].r = r;
	}
	sort(qq + 1, qq + cntq + 1, cmp);
	int lcur = 1, rcur = 0, tcur = 0;
	for (int i = 1; i <= cntq; i++)
	{
		while (lcur > qq[i].l) add(a[--lcur]);
		while (lcur < qq[i].l) del(a[lcur++]);
		while (rcur > qq[i].r) del(a[rcur--]);
		while (rcur < qq[i].r) add(a[++rcur]);
		while (tcur < qq[i].t) upd(i, ++tcur);
		while (tcur > qq[i].t) upd(i, tcur--);//增加t轴上的移动
		ans[qq[i].id] = sum;//得到最终答案
	}
	for (int i = 1; i <= cntq; i++) printf("%d\n", ans[i]);
	return 0;//结束&AC!
}

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

Ctrl AC

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值