题目

瞎猜一下我们只要\(n\)次询问就能确定出\(\{A_i\}\)来

感受一下大概是询问的区间越长代价就越小,比如询问\([l,n]\)或\([1,r]\)的代价肯定不会超过\([l,r]\)

所以大胆猜一下我们询问的只有一些前缀和后缀

首先我们肯定要询问一下\([1,n]\)的和,之后我们考虑顺次得到\(A_1\)到\(A_n\)的和

想得到\(A_1\),我们当然可以直接询问\([1,1]\),但是有\([1,n]\)的和我们询问\([2,n]\)的和也能得到\(A_1\)

同理我们想得到\(A_i\),我们可以直接询问\([1,i]\)的和,由于这个时候\(A_1\)到\(A_{i-1}\)的和都知道了,做一个差就能得到\(A_i\)的和;也可以询问\([i+1,n]\),也能得到\(A_i\)的和

于是我们维护一个前缀、后缀的\(\gcd\)显然哪个小选哪一个

这样复杂度是\(O(qn\log n)\)

考虑到一个序列的前后缀\(\gcd\)之会变化\(\log\)次,于是我们可以考虑用线段树来维护区间\(\gcd\),二分出每次变化的位置,一共要二分\(\log\)次,每次二分有一个\(\log\),查询前缀\(\gcd\)是\(\log^2\),四个\(\log\)显然啥都过不去

首先求区间\(\gcd\)就是没有必要的,我们只需要对于二分出来的这段区间查询一下这段区间里是否所有数都是当前前缀\(\gcd\)的倍数就好了,只有全都都是当前前缀\(\gcd\)的倍数前缀\(\gcd\)才不会发生变化,于是线段树上的查询变成了\(\log n\)

再来考虑有线段树为什么还要生硬地套一个二分,我们直接在线段树上二分即可,对于一个当前的端点\(l\),我们直接在线段树上查\([l,n]\)这段区间,\([l,n]\)在线段树上被分成了\(\log\)段区间,我们依次访问到这些区间;当我们访问到一段区间的时候,发现如果这个区间的\(\gcd\)不是当前前缀\(\gcd\)的倍数,那么说明这个变化点肯定在这个区间内,我们直接在这颗子树里二分;否则我们再去访问接下来的区间,这样复杂度是两个\(\log\)相加

于是整体的复杂度就是\(O(q\log^2n)\)

代码

#include<bits/stdc++.h>
#define re register
#define LL long long
inline int read() {
char c=getchar();int x=0;while(c<'0'||c>'9') c=getchar();
while(c>='0'&&c<='9') x=(x<<3)+(x<<1)+c-48,c=getchar();return x;
}
const int maxn=1e5+5;
inline int max(int a,int b) {return a>b?a:b;}
inline int min(int a,int b) {return a<b?a:b;}
int gcd(int a,int b) {return !b?a:gcd(b,a%b);}
struct Seg{int l,r,v;}b[2][105];
int n,Q,a[maxn];
int l[maxn<<2],r[maxn<<2],g[maxn<<2],pos[maxn];
inline void pushup(int i) {g[i]=gcd(g[i<<1],g[i<<1|1]);}
void build(int x,int y,int i) {
l[i]=x,r[i]=y;
if(x==y) {pos[x]=i;g[i]=a[x];return;}
int mid=x+y>>1;
build(x,mid,i<<1),build(mid+1,y,i<<1|1);
pushup(i);
}
void change(int x,int v) {
a[x]=v;x=pos[x];g[x]=v;x>>=1;
while(x) pushup(x),x>>=1;
}
int chk(int i,int v) {
if(l[i]==r[i]) return g[i]%v==0?l[i]:l[i]-1;
if(g[i<<1]%v==0) return chk(i<<1|1,v);
return chk(i<<1,v);
}
int find(int x,int y,int i,int v) {
if(x>r[i]||y<l[i]) return 0;
if(x<=l[i]&&y>=r[i]) {
if(g[i]%v==0) return r[i];
return chk(i,v);
}
int mid=l[i]+r[i]>>1;
int now=find(x,y,i<<1,v);
if(now!=mid&&now) return now;
return max(now,find(x,y,i<<1|1,v));
}
int calc(int i,int v) {
if(l[i]==r[i]) return g[i]%v==0?l[i]:l[i]+1;
if(g[i<<1|1]%v==0) return calc(i<<1,v);
return calc(i<<1|1,v);
}
int query(int x,int y,int i,int v) {
if(x>r[i]||y<l[i]) return 0;
if(x<=l[i]&&y>=r[i]) {
if(g[i]%v==0) return l[i];
return calc(i,v);
}
int mid=l[i]+r[i]>>1;
int now=query(x,y,i<<1|1,v);
if(now&&now!=mid+1) return now;
int t=query(x,y,i<<1,v);
if(t) return t;return now;
}
int main() {
n=read(),Q=read();
for(re int i=1;i<=n;i++) a[i]=read();
build(1,n,1);int x,v,tot,cnt,now,p;
while(Q--) {
x=read(),v=read();change(x,v);
tot=0;b[0][++tot].l=1;
while(b[0][tot].l<=n) {
b[0][tot].v=gcd(b[0][tot-1].v,a[b[0][tot].l]);
b[0][tot].r=find(b[0][tot].l,n,1,b[0][tot].v);++tot;
b[0][tot].l=b[0][tot-1].r+1;
}
--tot;
cnt=0;b[1][++cnt].r=n;
while(b[1][cnt].r>=1) {
b[1][cnt].v=gcd(b[1][cnt-1].v,a[b[1][cnt].r]);
b[1][cnt].l=query(1,b[1][cnt].r,1,b[1][cnt].v);++cnt;
b[1][cnt].r=b[1][cnt-1].l-1;
}
--cnt;now=1;p=1;
for(re int i=1;i<=cnt;i++) b[1][i].l--,b[1][i].r--;
LL ans=0;
while(p<n) {
while(b[0][now].r<p) now++;
while(b[1][cnt].r<p) --cnt;
ans+=1ll*min(b[0][now].v,b[1][cnt].v)*(min(b[0][now].r,b[1][cnt].r)-p+1);
p=min(b[0][now].r,b[1][cnt].r)+1;
}
printf("%lld\n",ans+b[0][tot].v);
}
return 0;
}

「LibreOJ NOI Round #2」签到游戏的更多相关文章

  1. LOJ576 「LibreOJ NOI Round #2」签到游戏

    题目 先进行一个转化: 每次花费\(\gcd\limits_{i=l+1}^rB_i\)的代价,可以连\((l,r)\)这一条边. 然后我们需要求\(0\sim n\)的最小生成树. 根据Kruska ...

  2. 「LibreOJ NOI Round #2」不等关系

    「LibreOJ NOI Round #2」不等关系 解题思路 令 \(F(k)\) 为恰好有 \(k\) 个大于号不满足的答案,\(G(k)\) 表示钦点了 \(k\) 个大于号不满足,剩下随便填的 ...

  3. LibreOJ #507. 「LibreOJ NOI Round #1」接竹竿

    二次联通门 : LibreOJ #507. 「LibreOJ NOI Round #1」接竹竿 /* LibreOJ #507. 「LibreOJ NOI Round #1」接竹竿 dp 记录一下前驱 ...

  4. 「LibreOJ NOI Round #1」验题

    麻烦的动态DP写了2天 简化题意:给树,求比给定独立集字典序大k的独立集是哪一个 主要思路: k排名都是类似二分的按位确定过程. 字典序比较本质是LCP下一位,故枚举LCP,看多出来了多少个独立集,然 ...

  5. #509. 「LibreOJ NOI Round #1」动态几何问题

    下面给出部分分做法和满分做法 有一些奇妙的方法可以拿到同样多的分数,本蒟蒻只能介绍几种常见的做法 如果您想拿18分左右,需要了解:质因数分解 如果您想拿30分左右,需要了解:一种较快的筛法 如果您想拿 ...

  6. #510. 「LibreOJ NOI Round #1」动态几何问题

    题目: 题解: 几何部分,先证明一下 \(KX = \sqrt{a},YL = \sqrt{b}\) 设左侧的圆心为 \(O\) ,连接 \(OK\) ,我们有 \(OK = r\). 然后有 \(r ...

  7. #507. 「LibreOJ NOI Round #1」接竹竿 dp

    题目: 题解: 我们考虑把每对花色相同的牌看作区间. 那么如果我们设 \(f_i\) 表示决策在 \([1,i]\) 内的最优答案. 那么有 \(f_i = max\{max\{(f_{j-1}+\s ...

  8. LOJ#510. 「LibreOJ NOI Round #1」北校门外的回忆(线段树)

    题面 传送门 题解 感谢\(@M\_sea\)的代码我总算看懂题解了-- 这个操作的本质就是每次把\(x\)的\(k\)进制最低位乘\(2\)并进位,根据基本同余芝士如果\(k\)是奇数那么最低位永远 ...

  9. LOJ 510: 「LibreOJ NOI Round #1」北校门外的回忆

    题目传送门:LOJ #510. 题意简述: 给出一个在 \(K\) 进制下的树状数组,但是它的实现有问题. 形式化地说,令 \(\mathrm{lowbit}(x)\) 为在 \(K\) 进制下的 \ ...

随机推荐

  1. ES6十大特性(转载CSDN)

    1. const和let关键字 const用于定义常量. let用于定义变量.但是JavaScript中不是已经有变量了吗? 是的,这很正确,但用var声明的变量具有函数作用域,并会被提升到顶部. 这 ...

  2. IIS身份验证和文件操作权限(三、ASP.NET模拟)

    一.配置ASP.NET模拟 注意:在配置[ASP.NET模拟]是还要配置[匿名身份验证]不知道为什么,有知道可以留言,互相学习 二.浏览站点 -- 操作文件 ①无操作权限 点击写入 ②有操作权限(特定 ...

  3. UDP 协议解析 - 1

    目录 1. 概述 2. UDP 的主要特点 3. UDP 的首部格式 3. UDP 校验和 3.1 伪首部 3.2 UDP 校验和计算方法 [参考文献] 1. 概述 用户数据报协议(UDP,User ...

  4. js获取url中的中文参数出现乱码

    解决方法 function getQueryString(key){ var reg = new RegExp("(^|&)"+key+"=([^&]*) ...

  5. Day 16 : Python 时间模块[time,]datetime[]及第三方模块的下载与安装

    在进行python程序开发时,除了可以使用python内置的标准模块外,还右许多第三方模块使用,可以在python官网找到. 在使用第三方模块时,需要下载并安装此模块,然后就可以使用标准模块一样导入并 ...

  6. JUC源码分析-集合篇(六)LinkedBlockingQueue

    JUC源码分析-集合篇(六)LinkedBlockingQueue 1. 数据结构 LinkedBlockingQueue 和 ConcurrentLinkedQueue 一样都是由 head 节点和 ...

  7. Spring Boot Restful WebAPI集成 OAuth2

    系统采用前后端分离的架构,采用OAuth2协议是很自然的事情. 下面开始实战,主要依赖以下两个组件: <dependency> <groupId>org.springframe ...

  8. python语言和R语言实现机器学习算法

    <转>机器学习系列(9)_机器学习算法一览(附Python和R代码)   转自http://blog.csdn.net/han_xiaoyang/article/details/51191 ...

  9. SpringMVC前后端参数交互

    Controller中使用JSON方式有多种 关键在于ajax请求是将数据以什么形式传递到后台 HTTP请求中: 如果是get请求,那么表单参数以name=value&name1=value1 ...

  10. Linux文件介绍

    Linux文件介绍 Linux 文件属性 可以通过命令ll+文件名,查看文件的具体属性 例如:ll syz.gz 1736706 -rw-r--r--. 1 root root 28 Oct 27 1 ...