原题传送门

二次离线莫队

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

我们需要考虑从(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. linux 安装mysql mysql-5.7.17-linux-glibc2.5-x86_64.tar.gz

    下边连接:https://www.cnblogs.com/zero-gg/p/8875598.html,大神的结晶啊,我就悄悄的留下来自己用,链接缺少的部分补充下: 1.包的下载方法:wget htt ...

  2. delphi开第二个进程报错cannot create file editorlineends.ttr

    网上说问题是windows系统补丁造成的,解决办法有卸补丁.装插件,还有自己搞个bat启动. 在网上看到最好的一个办法是: 把这个文件EditorLineEnds.ttr的后缀改为ttf,然后安装这个 ...

  3. MAC 隐藏功能

    finder 类: shift+ cmd + G  (去指定路径) cmd+↑ (返回) cmd+↓(打开当前选中的文件,如果没有选中的则去选中第一个) cmd+ o (打开当前选中的文件) 以下这些 ...

  4. 第07组 Beta版本演示

    组长博客:求戳 队名:摇光 成员 学号 姓名 组长 031702203 容慧珺 杨明哲 031702202 高星 杨明哲 031702307 黄森敏 杨明哲 031702308 朱丽辰 杨明哲 031 ...

  5. Fabric.js canvas 图形库

    1.github地址: https://github.com/fabricjs/fabric.js 2.简述 Fabric.js将canvas的编程变得简单.同时在canvas上添加了交互.交互包括: ...

  6. node 部署教程二

    转:https://www.cnblogs.com/yesyes/p/7168449.html 这篇文章主要介绍如何在服务端跑vuejs的项目,如果上一篇教程你成功输出了hello world,那这一 ...

  7. java8之Spliterator

    基本用法: import java.util.Arrays; import java.util.Spliterator; import java.util.stream.IntStream; publ ...

  8. RedisHelper Redis帮助类

    using StackExchange.Redis; using System; using System.Collections.Generic; using System.IO; using Sy ...

  9. python操作excel——openpyxl

    一.概述 python操作excel各个库对比:https://www.cnblogs.com/paul-liang/p/9187503.html 官方文档:https://openpyxl.read ...

  10. LeetCode:接雨水【42】

    LeetCode:接雨水[42] 题目描述 给定 n 个非负整数表示每个宽度为 1 的柱子的高度图,计算按此排列的柱子,下雨之后能接多少雨水. 上面是由数组 [0,1,0,2,1,0,1,3,2,1, ...