题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=4638

解题思路:

题意为询问一段区间里的数能组成多少段连续的数。先考虑从左往右一个数一个数添加,考虑当前添加了i - 1个数的答案是x,那么添加完i个数后的答案是多少?可以看出,是根据a[i]-1和a[i]+1是否已经添加而定的,如果a[i]-1或者a[i]+1已经添加一个,则段数不变,如果都没添加则段数加1,如果都添加了则段数减1。设v[i]为加入第i个数后的改变量,那么加到第x数时的段数就是sum{v[i]} (1<=i<=x}。仔细想想,若删除某个数,那么这个数两端的数的改变量也会跟着改变,这样一段区间的数构成的段数就还是他们的v值的和。将询问离线处理,按左端点排序后扫描一遍,左边删除,右边插入,查询就是求区间和。

以上摘自杭电的解题报告

标程:标程是从左边插入,从右边删除

 #include<cstdio>
#include<cstring>
#include<algorithm>
#include<iostream>
using namespace std;
const int N = ;
struct Q
{
int l,r,id;
}query[N];
bool cmp(Q x,Q y)
{
return x.r>y.r;
}
int a[N],c[N],d[N];
int lowbit(int x)
{
return x&(-x);
}
void up(int x,int v)
{
while(x<N)
{
c[x]+=v;
x+=lowbit(x);
}
}
int getsum(int x)
{
int r=;
while(x>)
{
r+=c[x];
x-=lowbit(x);
}
return r;
}
bool u[N];
int ret[N],ps[N];
int main()
{
int T,n,m,i,j;
scanf("%d",&T);
while(T--)
{
scanf("%d%d",&n,&m);
for(i=;i<=n;i++)
{
scanf("%d",&a[i]);
ps[a[i]]=i;
}
memset(c,,sizeof(c));
memset(d,,sizeof(d));
memset(u,,sizeof(u));
for(i=n;i>;i--)
{
if(u[a[i]-])d[i]++;
if(u[a[i]+])d[i]++;
if(d[i]==)
{
up(i,);
}
else if(d[i]==)
{
up(i,-);
}
u[a[i]]=true;
}
for(i=;i<m;i++)
{
scanf("%d%d",&query[i].l,&query[i].r);
query[i].id=i;
}
sort(query,query+m,cmp);
int j=n;
for(i=;i<m;i++)
{
while(j>query[i].r)
{
if(a[j]>&&ps[a[j]-]<j)
{
d[ps[a[j]-]]--;
up(ps[a[j]-],);
}
if(a[j]<n&&ps[a[j]+]<j)
{
d[ps[a[j]+]]--;
up(ps[a[j]+],);
}
j--;
}
ret[query[i].id]=getsum(query[i].r)-getsum(query[i].l-);
}
for(i=;i<m;i++)printf("%d\n",ret[i]);
}
return ;
}

我觉得看完解题思路后我还是不太明白,就模拟了一下样例。

3 1 2 5 4

得到的数组v 为 1 1 -1 1 0

这时查询任意1-k区间的段数=sum(vi)(1=<i<=k),都是对的···神奇···

在删除3后,3-1所在的位置为3,即v[3]=v[3]+1,3+1所在的位置为5。++v[5]。

v数组变为 1  1 0 1 0 ,查询2-i区间的段数=sum(vi)(2=<i<=k),同样是对的···更神奇···有兴趣的可以自己证明

我的代码:

 #include <cstdio>
#include <algorithm>
#define N 100005
using namespace std;
struct Node
{
int l,r,index;
} p[N];
int a[N],rank[N],ans[N],c[N];
int n,m; bool cmp(Node a,Node b)
{
return a.l < b.l;
}
int lowbit(int x)
{
return x&(-x);
}
void add(int x,int v)
{
while(x<N)
{
c[x] += v;
x += lowbit(x);
}
}
int sum(int x)
{
int s =;
while(x)
{
s += c[x];
x -= lowbit(x);
}
return s;
} void init()
{
scanf("%d%d",&n,&m);
rank[] =rank[n+] = n+;
for(int i=; i<=n; ++i)
{
scanf("%d",&a[i]);
rank[a[i]] = i;
}
for(int i=; i<m; ++i)
{
scanf("%d%d",&p[i].l,&p[i].r);
p[i].index = i;
}
}
int main()
{
// freopen("in.cpp","r",stdin);
int t;
scanf("%d",&t);
while(t--)
{
init();
memset(c, , sizeof(c) );
for(int i=; i<=n; ++i)
{
int d =;
if(rank[a[i]+] < i ) ++d;
if(rank[a[i]-] < i ) ++d;
if(d == )
add(i,);
else if(d == )
add(i,-);
}
sort(p,p+m,cmp);
rank[] = rank[n+] = ;
int j=;
for(int i=; i<m; ++i)
{
int t = p[i].l;
while(j < t)
{
if(rank[a[j]+] > j)
add(rank[a[j]+],);
if(rank[a[j]-] > j)
add(rank[a[j]-],);
++j;
}
ans[p[i].index] = sum(p[i].r)-sum(p[i].l-);
}
for(int i=; i<m; ++i)
printf("%d\n",ans[i]);
}
return ;
}

HDU 4638 树状数组 想法题的更多相关文章

  1. hdu 4638 树状数组 区间内连续区间的个数(尽可能长)

    Group Time Limit: 4000/2000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others) Total Subm ...

  2. hdu 4638 树状数组

    思路:将查询区间按右节点的升序排列,然后插入第i个数的时候,若nun[i]+1已经插入,那么就update(pre[num[i]+1],-1):pre[]表示的是该数的位置.同样若 num[i]-1存 ...

  3. hdu 1166 树状数组模板题

    #include<stdio.h> #include<string.h> #define N  51000 int  c[N],n; int number(int x) { r ...

  4. HDU 1166 敌兵布阵(线段树/树状数组模板题)

    敌兵布阵 Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Others) Total Submi ...

  5. st表树状数组入门题单

    预备知识 st表(Sparse Table) 主要用来解决区间最值问题(RMQ)以及维护区间的各种性质(比如维护一段区间的最大公约数). 树状数组 单点更新 数组前缀和的查询 拓展:原数组是差分数组时 ...

  6. hdu 4777 树状数组+合数分解

    Rabbit Kingdom Time Limit: 6000/3000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others) T ...

  7. 敌兵布阵 HDU - 1166 (树状数组模板题,线段树模板题)

    思路:就是树状数组的模板题,利用的就是单点更新和区间求和是树状数组的强项时间复杂度为m*log(n) 没想到自己以前把这道题当线段树的单点更新刷了. 树状数组: #include<iostrea ...

  8. 树状数组训练题1:弱弱的战壕(vijos1066)

    题目链接:弱弱的战壕 这道题似乎是vijos上能找到的最简单的树状数组题了. 原来,我有一个错误的思想,我的设计是维护两个树状数组,一个是横坐标,一个是纵坐标,然后读入每个点的坐标,扔进对应的树状数组 ...

  9. bzoj1103树状数组水题

    (卧槽,居然规定了修改的两点直接相连,亏我想半天) 非常水的题,用dfs序(而且不用重复,应该是直接规模为n的dfs序)+树状数组可以轻松水 收获:树状数组一遍A(没啥好骄傲的,那么简单的东西) #i ...

随机推荐

  1. 1.python+selenium利用cookie,跳过验证码直接登录

    方法1 在登录时,叫代码等待一段时间,然后手动输入验证码 # coding:utf-8 from selenium import webdriver import time url = 'http:/ ...

  2. Oracle 起诉 Google 事件

    最近,Google 和 Oracle 纠缠多年的“Java 侵权案”又有了新的判决结果.Google 在此次对决中败诉,并可能需要支付高达88亿美元的赔偿金.这个案件还引发关于 API(应用程序编程接 ...

  3. codeforces 555a//Case of Matryoshkas// Codeforces Round #310(Div. 1)

    题意:1-n的数字,大的在小的后面,以这种规则已经形成的几个串,现在要转为一个串,可用的操作是在末尾拆或添加,问要操作几次? 模拟了很久还是失败,看题解才知道是数学.看来这种只要结果的题,模拟很不合算 ...

  4. 『TensorFlow』DCGAN生成动漫人物头像_下

    『TensorFlow』以GAN为例的神经网络类范式 『cs231n』通过代码理解gan网络&tensorflow共享变量机制_上 『TensorFlow』通过代码理解gan网络_中 一.计算 ...

  5. 『Sklearn』特征向量化处理

    『Kaggle』分类任务_决策树&集成模型&DataFrame向量化操作 1 2 3 4 5 6 7 8 9 '''特征提取器''' from sklearn.feature_extr ...

  6. 关于floyd 打印路径的问题

    我们令    f[i][j]  表示从 i-->j的最短路上j前面的那个点. 显然初始化时  f[i][j]=i;  (这样的话先判断一下i是否能到达j好点) 更新条件时,当发现通过点k能使最短 ...

  7. Oracle11g温习-第一章 2、ORACLE 物理结构

    2013年4月27日 星期六 10:26 物理操作系统文件的集合.主要包括: 控制文件(参数文件init$ORACLE_SID.ora记录了控制文件的位置) 二进制文件,控制文件由参数control_ ...

  8. svn更新产生的异常

    同步svn时差生的错误如下:   同步 SVNStatusSubscriber 时报告了错误.1 中的 0 个资源已经同步. 同步 /tunnel14 时发生错误:Error getting stat ...

  9. <meta http-equiv="X-UA-Compatible" content="IE=7" />

    <meta http-equiv="X-UA-Compatible" content="IE=7" />意思是将IE8用IE7进行渲染,使网页在IE ...

  10. 063——VUE中vue-router之重定向redirct的使用

    <!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8&quo ...