[洛谷P5048][Ynoi2019模拟赛]Yuno loves sqrt technology III

题目大意:有$n(n\leqslant5\times10^5)$个数,$m(m\leqslant5\times10^5)$个询问,每个询问问区间$[l,r]$中众数的出现次数

题解:分块,设块大小为$S$,先可以预处理出两两块之间的众数出现次数,复杂度$O(\Big(\dfrac n S\Big)n)$。询问时先把答案设成整块内的答案,然后对两边剩下的最多$2S$个元素进行讨论。

难点在如何快速求出一个元素在区间内出现次数,先想到的是主席树,但是多了一个$\log_2$,并过不去。可以把每种数出现的位置用$vector$存下来,并且对每个数存一个它在$vector$中的位置,第$i$个数的位置是$ret_i$,假设现在的答案为$ans$,正在处理左边的多余元素,处理到第$i$个,可以看这个数值的$vector$中,第$ret_i+ans$位是否小于$r$,若小于,则这个数至少出现了$ans+1$次,更新答案。右边也是类似的。

发现$ans$最多自增$2S$次,所以一次查询的复杂度是$O(2S)$的,总复杂度为$O(2Sm+\Big(\dfrac n S\Big)n)$,当$S$略小于$\sqrt n$时最优(其实是我不怎么算)

卡点:$Ynoi$当然卡常啦

 

C++ Code:

#include <algorithm>
#include <cstdio>
#include <cctype>
#include <vector>

namespace std {
    struct istream {
#define M (1 << 24 | 3)
        char buf[M], *ch = buf - 1;
        inline istream() {
#ifndef ONLINE_JUDGE
            freopen("input.txt", "r", stdin);
#endif
            fread(buf, 1, M, stdin);
        }
        inline istream& operator >> (int &x) {
            while (isspace(*++ch));
            for (x = *ch & 15; isdigit(*++ch); ) x = x * 10 + (*ch & 15);
            return *this;
        }
#undef M
    } cin;
    struct ostream {
#define M (1 << 24 | 3)
        char buf[M], *ch = buf - 1;
        int w;
        inline ostream& operator << (int x) {
            if (!x) {
                *++ch = '0';
                return *this;
            }
            for (w = 1; w <= x; w *= 10);
            for (w /= 10; w; w /= 10) *++ch = (x / w) ^ 48, x %= w;
            return *this;
        }
        inline ostream& operator << (const char x) {*++ch = x; return *this;}
        inline ostream& operator << (const char *x) {
            while (*x) *this << *x++;
            return *this;
        }
        inline ~ostream() {
#ifndef ONLINE_JUDGE
            freopen("output.txt", "w", stdout);
#endif
            fwrite(buf, 1, ch - buf + 1, stdout);
        }
#undef M
    } cout;
}

#define maxn 500010
const int BSZ = 610, BNUM = maxn / BSZ + 10;

int cnt[maxn];
int MAX[BNUM][BNUM];
int L[maxn], R[maxn], bel[maxn];

int n, m, Bnum;
int w[maxn], v[maxn];

std::vector<int> list[maxn];
int ret[maxn];
int main() {
	std::cin >> n >> m;
	for (int i = 1; i <= n; i++) {
		std::cin >> w[i];
		v[i] = w[i];
		bel[i] = (i - 1) / BSZ + 1;
	}

	const int tot = (std::sort(v + 1, v + n + 1), std::unique(v + 1, v + n + 1) - v - 1);
	for (int i = 1; i <= n; i++) {
		w[i] = std::lower_bound(v + 1, v + tot + 1, w[i]) - v;
		ret[i] = list[w[i]].size();
		list[w[i]].push_back(i);
	}

	Bnum = bel[n];
	for (int i = 1; i <= Bnum; i++) {
		L[i] = (i - 1) * BSZ, R[i] = L[i] + BSZ - 1;
	}
	L[1] = 1, R[Bnum] = n;
	for (int i = 1; i <= Bnum; i++) {
		__builtin_memset(cnt, 0, sizeof cnt);
		int Max = 0, now = i;
		for (int j = L[i]; j <= n; j++) {
			cnt[w[j]]++;
			Max = std::max(Max, cnt[w[j]]);
			if (j == R[now]) {
				MAX[i][now] = Max;
				now++;
			}
		}
	}

	int ans = 0;
	while (m --> 0) {
		int l, r;
		std::cin >> l >> r;
		l ^= ans, r ^= ans;
		const int lb = bel[l], rb = bel[r];
		ans = 0;
		if (lb == rb) {
			for (int i = l; i <= r; i++) {
				const int W = w[i], sz = list[W].size(), pos = ret[i];
				while (pos + ans < sz && list[W][pos + ans] <= r) ans++;
			}
		} else {
			if (lb + 1 < rb) ans = MAX[lb + 1][rb - 1];
			for (int i = l; i <= R[lb]; i++) {
				const int W = w[i], sz = list[W].size(), pos = ret[i];
				while (pos + ans < sz && list[W][pos + ans] <= r) ans++;
			}
			for (int i = L[rb]; i <= r; i++) {
				const int W = w[i], pos = ret[i];
				while (pos >= ans && list[W][pos - ans] >= l) ans++;
			}
		}
		std::cout << ans << '\n';
	}
	return 0;
}

  

转载于:https://www.cnblogs.com/Memory-of-winter/p/10126812.html

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

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值