BZOJ-1878-HH的项链-SDOI2009

这篇博客详细介绍了SDOI2009年题目BZOJ-1878(HH的项链)的解题思路,包括离线树状数组统计和莫队算法两种方法。离线树状数组统计通过排序和更新避免了同种颜色重复计算,时间复杂度为O(nlogn);莫队算法则采用分块思想,以O(n√n)的时间复杂度解决了问题。文章还提供了完整的C++代码实现。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

描述

有n个位置(n ≤ 50000), 每个位置有一个颜色(颜色数 ≤ 200000).

询问[L, R]区间的颜色种数.


分析

  • 两种解法都做了一下
  • 1. 离线树状数组统计
  • HZWER:
  • 将所有询问按照左端点进行排序, 将所有颜色的第一个点x, 将a[x]++, 然后从左往右扫, 扫到一个点x, 将a[next[x]]++, 碰到一个询问l,r输出sum[r]-sum[l-1], 其中sum是a数组的前缀和, 求前缀和可以用树状数组.
  • 之所以这样做关键就是避免同种颜色计算多次, 这也是用树状数组解决问题的关键. 在L前的一个位置x, 如果next[x]在L之前, 用树状数组统计query(R)-query(L-1)时被减掉没有计算进去, 但之后一定会在结束对[L, R]的统计之前遇到这个next[x]; 如果next[x]在[L, R]的区间中, 那么更新next[x]之后query(R)-query(L-1)比原来大1, 也就是这个颜色被统计进去了, 而因为next[x]>L, 所以不会再遇到这个next[x]; 如果next[x]>R, 那么不会影响结果.
  • 好巧妙的方式
  • O(nlogn)
  • 2. 莫队算法
  • 分块的做法:
  • 将询问分 √n 块, 每一块内按R从小到大排序, 不同块之间按L从小到大排序. 然后用两个指针{p, q}扫描, 统计[p, q]区间内每个颜色的出现次数, 同时根据当前的答案统计对[L, R]区间的颜色种数造成的影响, 直到p==L&&q==R, 当前的答案就是[L, R]区间的答案.
  • 感性的理解是这样分块让指针p, q在统计某一询问移动的次数尽可能少. 至于为什么我觉得没必要证啦...
  • O(n√n)

#include 
  
  
   
   
#include 
   
   
#include 
    
    
     
     
#include 
     
     
      
      
using namespace std;

const int maxn = 50000 + 10;
const int maxm = 200000 + 10;

map
      
      
        IDcache; int cnt, cur, blocks; int L, R; int ans[maxm]; int A[maxn], c[maxn]; struct Question { int L, R, id; bool operator < (const Question& rhs) const { if(L/blocks == rhs.L/blocks) return R < rhs.R; return L < rhs.L; } } questions[maxm]; #define q questions[i] int id(int x) { if(IDcache.count(x)) return IDcache[x]; return IDcache[x] = ++cnt; } int main() { int n, m; scanf("%d", &n); for(int i = 1; i <= n; i++) { int x; scanf("%d", &x); A[i] = id(x); } scanf("%d", &m); for(int i = 0; i < m; i++) scanf("%d %d", &q.L, &q.R), q.id = i; blocks = (int)(sqrt(n) + 0.5); sort(questions, questions + m); for(int i = 0; i < m; i++) { while(L > q.L) if(++c[A[--L]] == 1) ++cur; while(R < q.R) if(++c[A[++R]] == 1) ++cur; while(L < q.L) if(--c[A[L++]] == 0) --cur; while(R > q.R) if(--c[A[R--]] == 0) --cur; ans[q.id] = cur; } for(int i = 0; i < m; i++) printf("%d\n", ans[i]); return 0; } #include 
       
         #include 
         #include 
         
           using namespace std; const int maxn = 50000 + 10; const int maxm = 200000 + 10; map 
          
            IDcache; int ID_cnt; int A[maxn]; int ans[maxm]; int first[maxn], next[maxn]; struct Question { int L, R, id; void init(int i) { id = i; scanf("%d %d", &L, &R); } bool operator < (const Question& rhs) const { if(L == rhs.L) return R < rhs.R; return L < rhs.L; } } questions[maxm]; struct BIT { int n; int c[maxn]; int lowbit(int x) { return x & (-x); } void modify(int x, int d) { for(; x <= n; x += lowbit(x)) c[x] += d; } int query(int x) { int ret = 0; for(; x > 0; x -= lowbit(x)) ret += c[x]; return ret; } } bit; int id(int x) { if(IDcache.count(x)) return IDcache[x]; return IDcache[x] = ++ID_cnt; } #define a A[i] #define q questions[i] int main() { int n, m; scanf("%d", &n); bit.n = n; for(int i = 1; i <= n; i++) { int x; scanf("%d", &x); A[i] = id(x); } for(int i = n; i >= 1; i--) { next[i] = first[a]; first[a] = i; } for(int i = 1; i <= ID_cnt; i++) if(first[i]) bit.modify(first[i], 1); scanf("%d", &m); for(int i = 0; i < m; i++) q.init(i); sort(questions, questions + m); int cur = 1; for(int i = 0; i < m; i++) { for(; cur < q.L; cur++) if(next[cur]) bit.modify(next[cur], 1); ans[q.id] = bit.query(q.R) - bit.query(q.L-1); } for(int i = 0; i < m; i++) printf("%d\n", ans[i]); return 0; } 
           
          
        
      
     
     
    
    
  
  


评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值