HDU 4777 Rabbit Kingdom 树状数组
分析:找到每一个点的左边离他最近的不互质数,记录下标(L数组),右边一样如此(R数组),预处理
这个过程需要分解质因数O(n*sqrt(n))
然后离线,按照区间右端点排序
然后扫一遍,对于当前拍好顺序的第i个询问,将所有小于r的点加入更新
更新的过程是这样的
(1)对于刚加入点x,树状数组L[x]位置+1 把这个定义为左更新
(2)对于所有R[i]=x的点,树状数组L[i]位置-1,i位置+1 把这个定义为右更新
(3)查询是询问区间 l->r的和
时间复杂度分析,因为每个点左更新,右更新各一次,所以单组测试用例是O(nlogn)的
下面来解释为啥这样更新
查看当前询问 l , r,对于所有小于 l 的点 i,它的所有更新(左更新和右更新)不会影响到树状数组区间 l 到 r 的 和
对于在l,r区间的点 i,如果 R[i]<=r,那么对于区间 l ->r 有 1 的贡献
如果 R[i]>r,对于这个点,只有左更新L[i]+1;
如果 L[i]>=l ,对于查询区间有 1 的贡献
如果 L[i]<l,对于查询区间没有贡献
不难发现,这样对于查询区间有贡献的点,都是会和别人打架的点
所以该查询的答案就是 查询区间长度 - 树状数组区间和
然后以下看代码
#include<cstdio>
#include<cstring>
#include<queue>
#include<cstdlib>
#include<algorithm>
#include<vector>
#include<cmath>
using namespace std;
typedef long long LL;
const int N=2e5;
const LL mod=1e9+;
int a[N+],c[N+];
int f[N+],h[N+],k[N+];
vector<int>g[N+];
vector<int>b[N+];
struct Que
{
int l,r,id;
bool operator<(const Que &e)const
{
return r<e.r;
}
}q[N+];
int n,m;
bool vis[N+];
int prime[N+],cnt,res[N+];
void add(int x,int t)
{
for(;x<=n+;x+=(x&(-x)))
c[x]+=t;
}
int get(int x)
{
int ans=;
for(;x>;x-=(x&(-x)))
ans+=c[x];
return ans;
}
int main()
{
for(int i=; i*i<=N; ++i)
{
if(vis[i])continue;
for(int j=i*i; j<=N; j+=i)
vis[j]=;
}
for(int i=; i<=N; ++i)
if(!vis[i])prime[cnt++]=i;
for(int i=; i<=N; ++i)
{
int t=i;
for(int j=; j<cnt&&prime[j]*prime[j]<=i; ++j)
{
if(t%prime[j])continue;
g[i].push_back(prime[j]);
while(t%prime[j]==)t/=prime[j];
if(t==)break;
}
if(!vis[t]&&t!=)
g[i].push_back(t);
}
while(~scanf("%d%d",&n,&m),n)
{
memset(f,,sizeof(f));
memset(c,,sizeof(c));
for(int i=; i<=n+; ++i)
{
scanf("%d",&a[i]);
h[i]=;
k[i]=n+;
}
for(int i=; i<=n+; ++i)
{
for(int j=; j<g[a[i]].size(); ++j)
{
int x=g[a[i]][j];
h[i]=max(h[i],f[x]);
f[x]=i;
}
}
for(int i=; i<=N; ++i)
f[i]=n+;
for(int i=n+; i>; --i)
{
for(int j=; j<g[a[i]].size(); ++j)
{
int x=g[a[i]][j];
k[i]=min(k[i],f[x]);
f[x]=i;
}
}
for(int i=;i<=m;++i)
scanf("%d%d",&q[i].l,&q[i].r),q[i].id=i;
sort(q+,q++m);
for(int i=;i<=n;++i)
b[i].clear();
for(int i=;i<=n+;++i)b[k[i]].push_back(i);
int x=;
for(int i=;i<=m;++i)
{
while(x<=n+&&x<=q[i].r+)
{
add(h[x],);
for(int j=;j<b[x].size();++j)
{
int y=b[x][j];
add(h[y],-);
add(y,);
}
++x;
}
int t=get(q[i].r+)-get(q[i].l);
res[q[i].id]=q[i].r-q[i].l+-t;
}
for(int i=;i<=m;++i)
printf("%d\n",res[i]); }
return ;
}
HDU 4777 Rabbit Kingdom 树状数组的更多相关文章
- HDU 4777 Rabbit Kingdom(树状数组)
HDU 4777 Rabbit Kingdom 题目链接 题意:给定一些序列.每次询问一个区间,求出这个区间和其它数字都互质的数的个数 #include <cstdio> #include ...
- HDU 4777 Rabbit Kingdom --容斥原理+树状数组
题意: 给一个数的序列,询问一些区间,问区间内与区间其他所有的数都互质的数有多少个. 解法: 直接搞有点难, 所谓正难则反,我们求区间内与其他随便某个数不互质的数有多少个,然后区间长度减去它就是答案了 ...
- HDU 3333 | Codeforces 703D 树状数组、离散化
HDU 3333:http://acm.hdu.edu.cn/showproblem.php?pid=3333 这两个题是类似的,都是离线处理查询,对每次查询的区间的右端点进行排序.这里我们需要离散化 ...
- HDU 3333 - Turing Tree (树状数组+离线处理+哈希+贪心)
题意:给一个数组,每次查询输出区间内不重复数字的和. 这是3xian教主的题. 用前缀和的思想可以轻易求得区间的和,但是对于重复数字这点很难处理.在线很难下手,考虑离线处理. 将所有查询区间从右端点由 ...
- HDU 3333 Turing Tree (树状数组)
题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=3333 题意就是询问区间不同数字的和. 比较经典的树状数组应用. //#pragma comment(l ...
- HDU 4325 Flowers(树状数组+离散化)
http://acm.hdu.edu.cn/showproblem.php?pid=4325 题意:给出n个区间和m个询问,每个询问为一个x,问有多少个区间包含了x. 思路: 因为数据量比较多,所以需 ...
- hdu 5775 Bubble Sort 树状数组
Bubble Sort 题目连接: http://acm.hdu.edu.cn/showproblem.php?pid=5775 Description P is a permutation of t ...
- HDU - 1541 Stars 【树状数组】
题目链接 http://acm.hdu.edu.cn/showproblem.php?pid=1541 题意 求每个等级的星星有多少个 当前这个星星的左下角 有多少个 星星 它的等级就是多少 和它同一 ...
- HDU 3854 Glorious Array(树状数组)
题意:给一些结点,每个结点是黑色或白色,并有一个权值.定义两个结点之间的距离为两个结点之间结点的最小权值当两个结点异色时,否则距离为无穷大.给出两种操作,一种是将某个结点改变颜色,另一个操作是询问当前 ...
随机推荐
- Python拷贝及多进程与类的问题
最近写python写的尤其不顺利,更多的debug,逐渐的深入,产出却比较少.应该是个瓶颈期,坚持坚持,厚着脸皮也要坚持下去. 0x00 拷贝问题 程序中涉及到多进程和协程,大致的模型是开了2+个进程 ...
- Transaction Log Truncation
--method 1-- ALTER DATABASE KIS_Sample3 SET RECOVERY SIMPLE ) ALTER DATABASE KIS_Sample3 SET RECOVER ...
- C# - Lambda 表达式
Lambda 表达式分为三个部分: 1 参数定义部分.参数是未类型化的,因此编译器可以根据上下文推断出他们的类型. 2 =>运算符,把Lambda表达式的参数与表达式体分开. 3 表达式体. d ...
- Xcode常用快捷键及代码格式刷(缩进)方法-b
Xcode版本:4.5.1 一.总结的常用命令: 隐藏xcode command+h 退出xcode command+q 关闭窗口 command+w 关闭所有窗口 command+option+w ...
- [BEC][hujiang] Lesson03 Unit1:Working life ---Grammar & Listening & Vocabulary
3 Working life p8 Grammar Gerund and infinitive(动名词和不定式) 一般而言: 1 动词后面接动名词还是不定式没有特定规则,主要取决于语言习 ...
- java中线程池的使用方法
1 引入线程池的原因 由于线程的生命周期中包括创建.就绪.运行.阻塞.销毁阶段,当我们待处理的任务数目较小时,我们可以自己创建几个线程来处理相应的任务,但当有大量的任务时,由于创建.销毁线程需要很大的 ...
- 安卓从业者应该关注:Android 6.0的运行时权限
Android 6.0,代号棉花糖,自发布伊始,其主要的特征运行时权限就很受关注.因为这一特征不仅改善了用户对于应用的使用体验,还使得应用开发者在实践开发中需要做出改变. 没有深入了解运行时权限的开发 ...
- [itint5]棋盘漫步
要注意dp[0][0]要初始化为1. int totalPath(vector<vector<bool> > &blocked) { int m = blocked.s ...
- Let's go! (Ubuntu下搭建Go语言环境)
自2009年Go语言发布以来,我一直在关注Go语言,如今Go语言已经发展到1.2版本,而且也收到越来越多的人关注这门语言.Go语言设计的目的就是为了解决执行数度快但是编译数度并不理想(如C++)以及编 ...
- Tomcat下server.xml中context介绍
conf/Context.xml是Tomcat公用的环境配置;若在server.xml中增加<Context path="/test" docBase="D:\te ...