传送门:http://www.lydsy.com/JudgeOnline/problem.php?id=2038
裸的莫队,注意要先移动右端点再移动左端点。
#include <cstdio> #include <algorithm> #include <cmath> const int maxn = 50005, maxm = 50005; int n, m, c[maxn], s[maxn], l, r, siz; long long fenzi[maxm], fenmu[maxm], tem, same; struct st { int l, r; int id; } q[maxm]; bool cmp(const st & aa, const st & ss) { if (aa.l / siz == ss.l / siz) { return aa.r < ss.r; } return aa.l / siz < ss.l / siz; } inline long long gcd(long long aa, long long ss) { long long tt; while (ss) { tt = aa; aa = ss; ss = tt % ss; } return aa; } int main(void) { //freopen("in.txt", "r", stdin); scanf("%d%d", &n, &m); siz = (int)sqrt((float)n + 0.5f); for (int i = 1; i <= n; ++i) { scanf("%d", c + i); } for (int i = 1; i <= m; ++i) { scanf("%d%d", &q[i].l, &q[i].r); q[i].id = i; } std::sort(q + 1, q + m + 1, cmp); for (int i = q[1].l; i <= q[1].r; ++i) { ++s[c[i]]; } for (int i = 0; i <= n; ++i) { same += (long long)s[i] * (long long)(s[i] - 1) >> 1; } l = q[1].l; r = q[1].r; fenmu[q[1].id] = (long long)(q[1].r - q[1].l + 1) * (long long)(q[1].r - q[1].l) >> 1; fenzi[q[1].id] = same; tem = gcd(fenmu[q[1].id], fenzi[q[1].id]); fenmu[q[1].id] /= tem; fenzi[q[1].id] /= tem; for (int i = 2; i <= m; ++i) { fenmu[q[i].id] = (long long)(q[i].r - q[i].l + 1) * (long long)(q[i].r - q[i].l) >> 1; while (r < q[i].r) { ++r; same += s[c[r]]; ++s[c[r]]; } while (r > q[i].r) { same += 1 - s[c[r]]; --s[c[r]]; --r; } while (l > q[i].l) { --l; same += s[c[l]]; ++s[c[l]]; } while (l < q[i].l) { same += 1 - s[c[l]]; --s[c[l]]; ++l; } fenzi[q[i].id] = same; tem = gcd(fenmu[q[i].id], fenzi[q[i].id]); fenmu[q[i].id] /= tem; fenzi[q[i].id] /= tem; } for (int i = 1; i <= m; ++i) { printf("%lld/%lld ", fenzi[i], fenmu[i]); } return 0; }