原题传送门

二次离线莫队

二次离线莫队的做法参考第十四分块(前体)的题解

我们需要考虑从(1,i)如何推到(1,i+1)

我们算过了a[i]的答案,考虑加入a[i]的贡献

我们需要在a[i]的所有约数上打标记,这个珂以直接暴力(因为约数是\(\sqrt n\)级别的)

我们还要考虑倍数的问题,当a[i]>32时,直接暴力更新倍数打标记;a[i]<=32时,设计一个状态s[1……32],每位1表示有这个约数,0表示没有这个约数,因为一共232个状态过多,所以每8个拆成一位,变成4个28=256状态,我们需要在0~255中第a[i]%8位为1的状态打标记即可

说的有可能不清楚,放一下代码

inline void update(register int x)
{
for(register int i=0;i<tot[x];++i)
++bs[fac[x][i]];
if(x<=32)
{
if(x<=8)
{
for(register int s=0;s<256;++s)
if(s>>(x-1)&1)
++s1[s];
}
else if(x<=16)
{
for(register int s=0;s<256;++s)
if(s>>(x-9)&1)
++s2[s];
}
else if(x<=24)
{
for(register int s=0;s<256;++s)
if(s>>(x-17)&1)
++s3[s];
}
else
{
for(register int s=0;s<256;++s)
if(s>>(x-25)&1)
++s4[s];
}
}
else
for(register int i=x;i<=100000;i+=x)
++ys[i];
}

我们考虑a[i+1]的答案,答案就是约数标记、倍数标记和状压标记之和。状压标记要单独计算(如下),c[x]就是一个长为32位的状态,表示x是否有1~32的约数

inline int calc(register unsigned int x)
{
x=c[x];
return s1[x&255]+s2[x>>8&255]+s3[x>>16&255]+s4[x>>24];
}

完整代码(有些我可能说不清,珂以结合代码理解)

#include <bits/stdc++.h>
#define N 100005
#define ll long long
#define getchar nc
using namespace std;
inline char nc(){
static char buf[100000],*p1=buf,*p2=buf;
return p1==p2&&(p2=(p1=buf)+fread(buf,1,100000,stdin),p1==p2)?EOF:*p1++;
}
inline int read()
{
register int x=0,f=1;register char ch=getchar();
while(ch<'0'||ch>'9'){if(ch=='-')f=-1;ch=getchar();}
while(ch>='0'&&ch<='9')x=(x<<3)+(x<<1)+ch-'0',ch=getchar();
return x*f;
}
inline void write(register ll x)
{
if(!x)putchar('0');if(x<0)x=-x,putchar('-');
static int sta[20];register int tot=0;
while(x)sta[tot++]=x%10,x/=10;
while(tot)putchar(sta[--tot]+48);
}
int n,m,a[N],blocksize,s1[256],s2[256],s3[256],s4[256];
struct query{
int l,r,id,bl;
}q[N];
inline bool cmp(register query a,register query b)
{
return a.bl!=b.bl?a.l<b.l:((a.bl&1)?a.r<b.r:a.r>b.r);
}
int fac[N][200],tot[N],ys[N],bs[N];
unsigned int c[N];
ll pres[N],sufs[N],ans[N],res[N];
struct node{
int l,r,id,op;
};
vector<node> L[N],R[N];
inline int calc(register unsigned int x)
{
x=c[x];
return s1[x&255]+s2[x>>8&255]+s3[x>>16&255]+s4[x>>24];
}
inline void update(register int x)
{
for(register int i=0;i<tot[x];++i)
++ys[fac[x][i]];
if(x<=32)
{
if(x<=8)
{
for(register int s=0;s<256;++s)
if(s>>(x-1)&1)
++s1[s];
}
else if(x<=16)
{
for(register int s=0;s<256;++s)
if(s>>(x-9)&1)
++s2[s];
}
else if(x<=24)
{
for(register int s=0;s<256;++s)
if(s>>(x-17)&1)
++s3[s];
}
else
{
for(register int s=0;s<256;++s)
if(s>>(x-25)&1)
++s4[s];
}
}
else
for(register int i=x;i<=100000;i+=x)
++bs[i];
}
inline void clear()
{
memset(s1,0,sizeof(s1));
memset(s2,0,sizeof(s2));
memset(s3,0,sizeof(s3));
memset(s4,0,sizeof(s4));
memset(ys,0,sizeof(ys));
memset(bs,0,sizeof(bs));
}
int main()
{
for(register int i=1;i<=100000;++i)
{
for(register int j=i;j<=100000;j+=i)
fac[j][tot[j]++]=i;
if(i<=32)
for(register int j=i;j<=100000;j+=i)
c[j]|=1u<<i-1;
}
n=read(),m=read();
blocksize=n/(sqrt(m*2/3)+1);
for(register int i=1;i<=n;++i)
a[i]=read();
for(register int i=1;i<=m;++i)
{
int l=read(),r=read();
q[i]=(query){l,r,i,(l-1)/blocksize+1};
}
sort(q+1,q+1+m,cmp);
for(register int i=1;i<=n;++i)
{
pres[i]=pres[i-1]+ys[a[i]]+bs[a[i]]+calc(a[i]);
update(a[i]);
}
clear();
for(register int i=n;i>=1;--i)
{
sufs[i]=sufs[i+1]+ys[a[i]]+bs[a[i]]+calc(a[i]);
update(a[i]);
}
clear();
q[0].l=1,q[0].r=0;
for(register int i=1;i<=m;++i)
{
int pl=q[i-1].l,pr=q[i-1].r,nl=q[i].l,nr=q[i].r;
ans[i]=pres[nr]-pres[pr]+sufs[nl]-sufs[pl];
if(nr>pr)
R[pl-1].push_back((node){pr+1,nr,i,-1});
if(nr<pr)
R[pl-1].push_back((node){nr+1,pr,i,1});
if(nl<pl)
L[nr+1].push_back((node){nl,pl-1,i,-1});
if(nl>pl)
L[nr+1].push_back((node){pl,nl-1,i,1});
}
for(register int i=1;i<=n;++i)
{
update(a[i]);
for(register int j=0;j<R[i].size();++j)
{
int l=R[i][j].l,r=R[i][j].r;
ll tmp=0;
for(register int k=l;k<=r;++k)
tmp+=ys[a[k]]+bs[a[k]]+calc(a[k]);
ans[R[i][j].id]+=tmp*R[i][j].op;
}
}
clear();
for(register int i=n;i>=1;--i)
{
update(a[i]);
for(register int j=0;j<L[i].size();++j)
{
int l=L[i][j].l,r=L[i][j].r;
ll tmp=0;
for(register int k=l;k<=r;++k)
tmp+=ys[a[k]]+bs[a[k]]+calc(a[k]);
ans[L[i][j].id]+=tmp*L[i][j].op;
}
}
for(register int i=1;i<=m;++i)
ans[i]+=ans[i-1],res[q[i].id]=ans[i]+q[i].r-q[i].l+1;
for(register int i=1;i<=m;++i)
write(res[i]),puts("");
return 0;
}

【题解】Luogu P5398 [Ynoi2018]GOSICK的更多相关文章

  1. 洛谷P5398 [Ynoi2018]GOSICK(二次离线莫队)

    题面 传送门 题解 维包一生推 首先请确保您会二次离线莫队 那么我们现在的问题就是怎么转移了,对于\(i\)和前缀\([1,r]\)的贡献,我们拆成\(b_i\)和\(c_i\)两部分,其中\(b_i ...

  2. [题解] Luogu P5446 [THUPC2018]绿绿和串串

    [题解] Luogu P5446 [THUPC2018]绿绿和串串 ·题目大意 定义一个翻转操作\(f(S_n)\),表示对于一个字符串\(S_n\), 有\(f(S)= \{S_1,S_2,..., ...

  3. 题解 Luogu P2499: [SDOI2012]象棋

    关于这道题, 我们可以发现移动顺序不会改变答案, 具体来说, 我们有以下引理成立: 对于一个移动过程中的任意一个移动, 若其到达的位置上有一个棋子, 则该方案要么不能将所有棋子移动到最终位置, 要么可 ...

  4. 题解 luogu P1144 【最短路计数】

    本蒟蒻也来发一次题解第一篇请见谅 这个题有几个要点 1.无向无权图,建图的时候别忘记建来回的有向边[因此WA掉1次 2.无权嘛,那么边长建成1就好了2333333 3.最短路采用迪杰斯特拉(别忘用堆优 ...

  5. 题解 Luogu P1110 【[ZJOI2007]报表统计】

    感谢 @cmy962085349 提供的hack数据,已经改对了. 先声明,我好像是题解里写双$fhq$ $treap$里唯一能过的...(最后两个点啊) 思路:首先看题目,$MIN_GAP_SORT ...

  6. 题解 Luogu P3370

    讲讲这题的几种做法: 暴力匹配法 rt,暴力匹配,即把字符串存起来一位一位判相等 时间复杂度$ O(n^2·m) $ 再看看数据范围 \(n\le10^5,m\le10^3\) 当场爆炸.当然有暴力分 ...

  7. 题解 Luogu P3623 [APIO2008]免费道路

    [APIO2008]免费道路 题目描述 新亚(New Asia)王国有 N 个村庄,由 M 条道路连接.其中一些道路是鹅卵石路,而其它道路是水泥路.保持道路免费运行需要一大笔费用,并且看上去 王国不可 ...

  8. [题解]luogu P4116 Qtree3

    终于来到了Qtree3, 其实这是Qtree系列中最简单的一道题,并不需要线段树, 只要树链剖分的一点思想就吼了. 对于树链剖分剖出来的每一根重链,在重链上维护一个Set就好了, 每一个Set里存的都 ...

  9. 【洛谷5398】[Ynoi2018]GOSICK(二次离线莫队)

    题目: 洛谷 5398 当我刚学莫队的时候,他们告诉我莫队能解决几乎所有区间问题: 现在,当我发现一个区间问题似乎难以用我所了解的莫队解决的时候,他们就把这题的正解叫做 XXX 莫队.--题记 (以上 ...

随机推荐

  1. 在GitHub上使用Hexo 搭建自己的博客

    1.下载Node.js安装文件(现在电脑基本都是64位的,我就放64位的下载地址):https://nodejs.org/dist/v8.9.4/node-v8.9.4-x64.msi 或者自行到官网 ...

  2. Vector(动态数组)怎么用咧↓↓↓

    定义方式:vector<int> a; //二维vector<int>a[100] 在末尾压入容器:a.push_back(x);//二维 a[i].push_back(x) ...

  3. 移动端点击事件兼容问题,在pc端可以点,在手机上不可以点

    ms-click="showCodeExplain()" onClick="javascript:;" 在点击事件后面加上onClick="javas ...

  4. 2019暑假Java学习笔记(三)

    目录 面向对象 对象 构造方法 引用与对象实例 static final 封装 this 继承 super 方法重载与重写 多态 抽象类 接口 内部类 成员内部类 静态内部类 局部内部类 匿名内部类 ...

  5. 为什么GPU不能代替CPU?

    gpu就是并行处理强大, cpu很多功能gpu都没有. 什么指令流水化, 多进程管理之类的. gpu没有多少自主处理指令的能力, 基本是指令靠cpu 计算靠gpu.GPU工作原理是cpu 处理指令,遇 ...

  6. python 玩爬虫安装了一大堆第三方库

    之前就听说过爬虫,感觉很复杂的样子,但是看到python代码很简短.由于本机已经安装了python2.7 所以就拿来py 文件跑一下想看看效果. 结果各种代码错误.然后根据每个错误去下载对应的依赖项. ...

  7. Xamarin图表开发基础教程(7)OxyPlot框架

    Xamarin图表开发基础教程(7)OxyPlot框架 Xamarin.Forms中使用OxyPlot框架 在Xamarin. Forms平台上实现图表显示需要完成以下的步骤: 1.添加OxyPlot ...

  8. 【WebSocket】WebSocket快速入门

    WebSocket介绍 WebSocket 是 HTML5 开始提供的一种在单个 TCP 连接上进行全双工通讯的协议. WebSocket 使得客户端和服务器之间的数据交换变得更加简单,允许服务端主动 ...

  9. [译]使用to_dict将pandas.DataFrame转换为Python中的字典列表

    pandas.DataFrame.to_json返回的是JSON字符串,不是字典. 可以使用to_dict进行字典转换. 使用orient指定方向. >>> df col1 col2 ...

  10. MySQL之Xtrabackup使用

    Xtrabackup对使用innodb存储引擎的mysql数据库进行备份时,不会影响数据库的读写操作(网上是这么说的,我还没验证过) 1.安装yum源 yum install https://repo ...