P1972 [SDOI2009]HH的项链[离线+树状数组/主席树/分块/模拟]

时间:2022-09-06 14:57:19

题目背景

题目描述

HH 有一串由各种漂亮的贝壳组成的项链。HH 相信不同的贝壳会带来好运,所以每次散步完后,他都会随意取出一段贝壳,思考它们所表达的含义。HH 不断地收集新的贝壳,因此,他的项链变得越来越长。有一天,他突然提出了一个问题:某一段贝壳中,包含了多少种不同的贝壳?这个问题很难回答……因为项链实在是太长了。于是,他只好求助睿智的你,来解决这个问题。

输入输出格式

输入格式:

第一行:一个整数N,表示项链的长度。

第二行:N 个整数,表示依次表示项链中贝壳的编号(编号为0 到1000000 之间的整数)。

第三行:一个整数M,表示HH 询问的个数。

接下来M 行:每行两个整数,L 和R(1 ≤ L ≤ R ≤ N),表示询问的区间。

输出格式:

M 行,每行一个整数,依次表示询问对应的答案。

输入输出样例

输入样例#1:

6
1 2 3 4 3 5
3
1 2
3 5
2 6

输出样例#1:

2
2
4

说明

数据范围:

对于100%的数据,N <= 500000,M <= 500000。

解析:

我居然为了这道题去学了主席树???结果发现树状数组秒解???主席树还MLE???

我真是超级蒻啊,居然真不会写。。。

口胡请见谅。。。


正解:离线+树状数组/主席树/分块/模拟(雾

我们分析题目,会发现实际上题目就是要我们求一个区间内的不同的数。

我们有一个巧妙的模板解法,可以用不同的数据结构实现:

以每个不同的位置建立数据结构。

我们设一个数组\(a[i]=num\)表示在\(i\)位置上,总共有\(num\)种不同的数。

对于修改:

对于任意的一个数列,我们按照顺序将这个数列从\(1-n\)的位置上的数依次加入\(a\)数组,我们会发现,实际上我们只需维护在当前插入位置\(i\)下,\(1-i\)中各个不同的数出现的最后的位置。

对于询问:

题解有一个\(dalao\)总结的精辟:对于若干个询问的区间[l,r],如果他们的r都相等的话,那么项链中出现的同一个数字,一定是只关心出现在最右边的那一个的

当然这种做法相当于告诉我们要边处理插入边记录答案,而不能处理完后再记录答案。

对于数组\(a\),数的位置是移动的。

举个例子:

\(1~2~1~3~4\)

这个数列中,我们的\(a\)数组原先是

\(0~0~0~0~0\)

\(a\)数组加入位置\(1\)的\(1\)后

\(1~0~0~0~0\)

加入位置\(2\)的\(2\)后

\(1~1~0~0~0\)

加入位置\(3\)的\(1\)后

\(0~1~1~0~0\)

就到这里为止,我们就能得出结论:对于一个区间内相同的数字,我们只用关心它最后出现的位置,这样相当于我们只允许所有相同的数字在这个区间内出现一次。

这样我们就可以得出在某一个区间不同的数的总数了,就是插入\(i\)位置的数时\(a\)数组中所有\(1\)的个数。

参考代码:

AC 树状数组

#include<iostream>
#include<cstdio>
#include<cstring>
#include<cmath>
#include<algorithm>
#include<string>
#include<cstdlib>
#include<queue>
#include<vector>
#define INF 0x3f3f3f3f
#define PI acos(-1.0)
#define N 500010
#define MOD 2520
#define E 1e-12
using namespace std;
int ask[N<<1],n,m,last[N<<1],a[N<<1],ans[N<<1];
struct rec{
int l,r;
int id;
}q[N];
int query(int x)
{
int ans=0;
for(;x;x-=x&-x) ans+=ask[x];
return ans;
}
void add(int x,int y)
{
for(;x<=n;x+=x&-x) ask[x]+=y;
}
bool cmp(rec a,rec b)
{
return a.r<b.r;
}
int main()
{
scanf("%d",&n);
for(int i=1;i<=n;i++) scanf("%d",&a[i]);
scanf("%d",&m);
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,cmp);
int next=1;
for(int i=1;i<=m;i++){
for(int j=next;j<=q[i].r;j++){
int v=a[j];
if(last[v])
add(last[v],-1);
add(j,1);
last[v]=j;
}
next=q[i].r+1;
ans[q[i].id]=query(q[i].r)-query(q[i].l-1);
}
for(int i=1;i<=m;i++) printf("%d\n",ans[i]);
return 0;
}

无耻的贴上MLE2个点的主席树:

#include<iostream>
#include<cstdio>
#include<cstring>
#include<cmath>
#include<algorithm>
#include<string>
#include<cstdlib>
#include<queue>
#include<vector>
#define INF 0x3f3f3f3f
#define PI acos(-1.0)
#define N 500010
#define MOD 2520
#define E 1e-12
#define ri register int
using namespace std;
struct tree{
int lc,rc;
int sum;
}t[N<<6];
int tot,n,m,root[N*20],last[N*20];
inline int read()
{
int f=1,x=0;char c=getchar();
while(c<'0'||c>'9'){if(c=='-')f=-1;c=getchar();}
while(c>='0'&&c<='9'){x=x*10+c-'0';c=getchar();}
return x*f;
}
inline int build(int l,int r)
{
int p=++tot;
if(l==r) return p;
int mid=(l+r)>>1;
t[p].lc=build(l,mid);
t[p].rc=build(mid+1,r);
return p;
}
inline int change(int pre,int v,int l,int r,int val)
{
int p=++tot;
t[p].lc=t[pre].lc,t[p].rc=t[pre].rc,t[p].sum=t[pre].sum+v;
if(l==r) return p;
int mid=(l+r)>>1;
if(val<=mid) t[p].lc=change(t[pre].lc,v,l,mid,val);
else t[p].rc=change(t[pre].rc,v,mid+1,r,val);
return p;
}
inline int query(int x,int pre,int l,int r)
{
if(l==r) return t[pre].sum;
int mid=(l+r)>>1;
if(x<=mid) return query(x,t[pre].lc,l,mid)+t[t[pre].rc].sum;
else return query(x,t[pre].rc,mid+1,r);
}
int main()
{
n=read();
root[0]=t[0].lc=t[0].rc=t[0].sum;
for(ri i=1;i<=n;i++){
int x;
x=read();
if(last[x]){
int t=change(root[i-1],-1,1,n,last[x]);
root[i]=change(t,1,1,n,i);
}
else{
root[i]=change(root[i-1],1,1,n,i);
}
last[x]=i;
} m=read();
for(ri i=1;i<=m;i++)
{
int l,r;
l=read(),r=read();
printf("%d\n",query(l,root[r],1,n));
}
return 0;
}

P1972 [SDOI2009]HH的项链[离线+树状数组/主席树/分块/模拟]的更多相关文章

  1. 【bzoj3744】Gty的妹子序列 分块&plus;树状数组&plus;主席树

    题目描述 我早已习惯你不在身边, 人间四月天 寂寞断了弦. 回望身后蓝天, 跟再见说再见…… 某天,蒟蒻Autumn发现了从 Gty的妹子树(bzoj3720) 上掉落下来了许多妹子,他发现 她们排成 ...

  2. zoj2112 树状数组&plus;主席树 区间动第k大

    Dynamic Rankings Time Limit: 10000MS   Memory Limit: 32768KB   64bit IO Format: %lld & %llu Subm ...

  3. BZOJ&lowbar;1901&lowbar;Zju2112 Dynamic Rankings&lowbar;树状数组&plus;主席树

    BZOJ_1901_Zju2112 Dynamic Rankings_树状数组+主席树 题意: 给定一个含有n个数的序列a[1],a[2],a[3]……a[n],程序必须回答这样的询问:对于给定的i, ...

  4. 【bzoj1146】&lbrack;CTSC2008&rsqb;网络管理Network 倍增LCA&plus;dfs序&plus;树状数组&plus;主席树

    题目描述 M公司是一个非常庞大的跨国公司,在许多国家都设有它的下属分支机构或部门.为了让分布在世界各地的N个部门之间协同工作,公司搭建了一个连接整个公司的通信网络.该网络的结构由N个路由器和N-1条高 ...

  5. BZOJ&lowbar;2120&lowbar;数颜色&lowbar;Set&plus;树状数组&plus;主席树

    BZOJ_2120_数颜色_Set+树状数组+主席树 Description 墨墨购买了一套N支彩色画笔(其中有些颜色可能相同),摆成一排,你需要回答墨墨的提问.墨墨会像你发布如下指令: 1. Q L ...

  6. &lbrack;luoguP1972&rsqb; &lbrack;SDOI2009&rsqb;HH的项链(莫队 &vert;&vert; 树状数组 &vert;&vert; 主席树)

    传送门 莫队基础题,适合我这种初学者. 莫队是离线算法,通常不带修改,时间复杂度为 O(n√n) 我们要先保证通过 [ l , r ] 求得 [ l , r + 1 ] , [ l , r - 1 ] ...

  7. 【BZOJ】1146&colon; &lbrack;CTSC2008&rsqb;网络管理Network(树链剖分&plus;线段树套平衡树&plus;二分 &sol; dfs序&plus;树状数组&plus;主席树)

    http://www.lydsy.com/JudgeOnline/problem.php?id=1146 第一种做法(时间太感人): 第二种做法(rank5,好开心) ================ ...

  8. BZOJ 2743&colon; &lbrack;HEOI2012&rsqb;采花 &lbrack;树状数组 &vert; 主席树&rsqb;

    题意: 查询区间中出现次数$>2$的颜色个数 一眼主席树,区间中$l \le last[i] \le r$的个数减去$l \le last[last[i]] \le r$的个数,搞两颗主席树来做 ...

  9. HDU 3333 Turing Tree(树状数组&sol;主席树)

    题意 给定一个长度为 \(n​\) 的序列,\(m​\) 个查询,每次查询区间 \([L,R]​\) 范围内不同元素的和. \(1\leq T \leq 10\) \(1 \leq n\leq 300 ...

随机推荐

  1. iOS从零开始学习直播之音频1&period;播放本地音频文件

      现在直播越来越火,俨然已经成为了下一个红海.作为一个资深码农(我只喜欢这样称呼自己,不喜欢别人这样称呼我),我必须赶上时代的潮流,开始研究视频直播.发现视屏直播类的文章上来就讲拉流.推流.采集.美 ...

  2. Apache设置页面认证(原创贴-转载请注明出处)

    ================写在前面的话============== 1.本试验使用的apache版本是2.4.24 场景描述:网站后台管理页面比较重要,不应该任何人都让访问,所以对后台页面做认证 ...

  3. Entity Framework学习笔记&lpar;六&rpar;----使用Lambda查询Entity Framework&lpar;1&rpar;

    请注明转载地址:http://www.cnblogs.com/arhat 在前几章中,老魏一直使用Linq来查询Entity Framework.但是老魏感觉,如果使用Linq的话,那么Linq的返回 ...

  4. CoreDate的使用

    勾选 xcode的 CoreDate会帮我们自动创建 CoreData 但是我们通常不那样使用,通常把 CoreDate 在单利类中创建, // // ZYDAO.h // StoryboardTes ...

  5. C&num;多线程及GDI&lpar;Day 23&rpar;

       又来到了总结知识的时间了,今天又学了一些新的知识,是多线程和GDI的一些运用. 理论: 在学习多线程之前,首先要了解一下什么是进程? 进程:(关键字Process)进程是一个具有一定独立功能的程 ...

  6. 通过JSP&plus;servlet实现文件上传功能

    在TCP/IP中,最早出现的文件上传机制是FTP.它将文件由客户端到服务器的标准机制. 但是在JSP中不能使用FTP来上传文件,这是有JSP的运行机制所决定的. 通过为表单元素设置Method=&qu ...

  7. 还在问跨域?本文记录js跨域的多种实现实例

    前言 众所周知,受浏览器同源策略的影响,产生了跨域问题,那么我们应该如何实现跨域呢?本文记录几种跨域的简单实现 前期准备 为了方便测试,我们启动两个服务,10086(就是在这篇博客自动生成的项目,请戳 ...

  8. Redis 保护模式

    默认 redis 启用了保护模式,即如果是远程链接不能进行 CRUD 等操作,如果进行该操作报错如下 (error) DENIED Redis is running in protected mode ...

  9. OS &plus; CentOS 7 &sol; firefox

    s 一.安装firefox二.缺少so依赖如下步骤操作 1.缺少so依赖:下载firefox依赖so文件:libgtk-3.so.0.1400.13.libgdk-3.so.0.1400.13.lib ...

  10. maven 结合mybaits整合框架,打包时mapper&period;xml文件,mapper目录打不进war包去问题

    首先,来看下MAVENx项目标准的目录结构: 一般情况下,我们用到的资源文件(各种xml,properites,xsd文件等)都放在src/main/resources下面,利用maven打包时,ma ...