题解【bzoj2038 [2009国家集训队]小Z的袜子(hose)】

Description

\(m\) 个询问,每次给出一个区间,求从这个区间中取出两个数使得它们同色的概率。

\(n,m,a_i \leq 50000\)

Solution

莫队模板题

最后的概率是 选的颜色相同的方案数 / 区间长度 * (区间长度 - 1),显然,只需要维护方案数。

问题化为知道 \([l,r]\) 的情况下,如何快速算出 \([l \pm 1, r], [l, r \pm 1]\) 的方案数

如果加入一个数 \(x\),则方案数增加了 $cnt_x * (cnt_x+1) - cnt_x * (cnt_x-1) = 2 * cnt_x $

如果删除一个数 \(x\),则方案数减少了 \(cnt_x * (cnt_x - 1) - (cnt_x - 1) * (cnt_x - 2) = 2 * (cnt_x-1)\)

其中 \(cnt_x\) 表示 \(x\) 出现的次数。然后就做完了(

注意特判 \(l = r\)

Code

#include <bits/stdc++.h>
#define ll long long
using namespace std;
const int N = 500100; 
int n, m, a[N], cnt[N], blo;  
ll now, ans1[N], ans2[N];
struct node {
  int l, r, id; 
  bool operator < (const node &x) const {
    return l / blo == x.l / blo ? r < x.r : l / blo < x.l / blo;  
  }
} Q[N]; 
inline ll gcd(ll a, ll b) { return !b ? a : gcd(b, a % b); }
inline void add(int x) { 
  now += 2ll * cnt[a[x]]; 
  cnt[a[x]]++;
}
inline void del(int x) { 
  if(cnt[a[x]] > 0) {
    now -= 2ll * cnt[a[x]] - 2;
    cnt[a[x]]--; 
  }
}
int main() {
  scanf("%d %d", &n, &m); blo = sqrt(m);  
  for(int i = 1; i <= n; i++) scanf("%d", &a[i]); 
  for(int i = 1; i <= m; i++) {
    scanf("%d %d", &Q[i].l, &Q[i].r); Q[i].id = i; 
  } sort(Q + 1, Q + m + 1); 
  int L = 0, R = 0; 
  for(int i = 1; i <= m; i++) {
    int l = Q[i].l, r = Q[i].r;
    if(l == r) {
      ans1[Q[i].id] = 0, ans2[Q[i].id] = 1;
      continue ; 
    }
    while(L > l) add(--L); 
    while(R < r) add(++R); 
    while(L < l) del(L++); 
    while(R > r) del(R--); 
    ans1[Q[i].id] = now; 
    ans2[Q[i].id] = 1ll * (r - l + 1) * (r - l); 
  }
  for(int i = 1; i <= m; i++) {
    if(!ans1[i]) ans2[i] = 1; 
    else {
      int G = gcd(ans1[i], ans2[i]); 
      ans1[i] /= G, ans2[i] /= G; 
    } printf("%lld/%lld\n", ans1[i], ans2[i]); 
  }
  return 0; 
}
posted @ 2018-12-22 16:28  AcFunction  阅读(257)  评论(0编辑  收藏  举报