【CodeForces】671 C. Ultimate Weirdness of an Array
【题目】C. Ultimate Weirdness of an Array
【题意】给定长度为n的正整数序列,定义一个序列的价值为max(gcd(ai,aj)),1<=i<j<=n,定义f(i,j)为移除序列i~j后剩余序列的价值,求Σf(i,j)。1<=n,ai<=2*10^5。
【算法】数论+线段树
【题解】要求所有区间的f(i,j),转化为,记ans[i]表示f(l,r)=i的区间数量,则ANS=Σi*ans[i],i=1~mx,mx=max(ai)。
求解ans[i]不方便,记h[i]表示f(l,r)<=i的区间数量,则ans[i]=h[i]-h[i-1],显然h[i]单调不减。
考虑x=mx时,h[x]=n*(n+1)/2(即所有区间)。当x=mx-1时,设数列中mx的倍数有k个,就需要满足所选区间包含至少k-1个mx的倍数。随着x的减少,逐个将不满足的区间删除,就可以得到所有h[x]。
如何实现删除不满足的区间?
记v[i]表示数列中所有i的倍数的位置,用vector存储为v[i][k],这一步甚至可以用O(n√n)的分解素因数实现。
如果(l,r)合法,那么(l,R),R>r也一定合法。所以记next[i]表示L=i,R>=next[i]的区间均合法,初始next[i]=i。
对于x,h[x]=Σ(n-next[i]+1),i=1~n。
每次x减少,需要删除不满足x的区间时,假设x的倍数的位置为b1,b2……bk,需要进行以下3种操作:
1.p>b2,next[p]=n+1
2.b1<p<=b2,next[p]=max(next[p],bk)
3.1<=p<=b1,next[p]=max(next[p],bk-1)。
容易发现,next[i]是一个单调不减的数组,那么以上3个操作都可以用线段树的区间覆盖实现,答案用区间求和实现。(套路:线段树区间取max只能在单调的前提下通过区间覆盖实现)
具体而言,线段树参数需要维护max,min,delta,sum,当min>=x时直接return,当max<=x时直接修改。
复杂度O(n log n)。
- #include<cstdio>
- #include<cstring>
- #include<cctype>
- #include<vector>
- #include<algorithm>
- #define ll long long
- using namespace std;
- int read(){
- char c;int s=,t=;
- while(!isdigit(c=getchar()))if(c=='-')t=-;
- do{s=s*+c-'';}while(isdigit(c=getchar()));
- return s*t;
- }
- const int maxn=;
- struct tree{int l,r,mins,maxs,delta;ll sum;}t[maxn*];
- int n,a[maxn];
- ll h[maxn];
- vector<int>v[maxn];
- int min(int a,int b){return a<b?a:b;}
- int max(int a,int b){return a>b?a:b;}
- void up(int k){
- t[k].sum=t[k<<].sum+t[k<<|].sum;
- t[k].mins=min(t[k<<].mins,t[k<<|].mins);
- t[k].maxs=max(t[k<<].maxs,t[k<<|].maxs);
- }
- void modify(int k,int x){t[k].delta=x;t[k].mins=t[k].maxs=x;t[k].sum=1ll*x*(t[k].r-t[k].l+);}
- void down(int k){
- if(t[k].delta){
- modify(k<<,t[k].delta);
- modify(k<<|,t[k].delta);
- t[k].delta=;
- }
- }
- void build(int k,int l,int r){
- t[k].l=l;t[k].r=r;t[k].delta=;
- if(l==r){
- t[k].maxs=t[k].mins=t[k].sum=l;
- }
- else{
- int mid=(l+r)>>;
- build(k<<,l,mid);
- build(k<<|,mid+,r);
- up(k);
- }
- }
- void fix(int k,int l,int r,int x){
- if(l>r||t[k].mins>=x)return;
- if(l<=t[k].l&&t[k].r<=r&&t[k].maxs<=x){modify(k,x);return;}
- down(k);
- int mid=(t[k].l+t[k].r)>>;
- if(l<=mid)fix(k<<,l,r,x);
- if(r>mid)fix(k<<|,l,r,x);
- up(k);
- }
- int main(){
- n=read();
- int mx=;
- for(int i=;i<=n;i++){
- a[i]=read();
- for(int j=;j*j<=a[i];j++)if(a[i]%j==){
- v[j].push_back(i);
- if(j*j!=a[i])v[a[i]/j].push_back(i);
- }
- mx=max(mx,a[i]);
- }
- build(,,n);
- for(int i=mx+;i>=;i--){
- if(v[i].size()>=){
- fix(,v[i][]+,n,n+);
- fix(,v[i][]+,v[i][],v[i][v[i].size()-]);
- fix(,,v[i][],v[i][v[i].size()-]);
- }
- h[i]=1ll*n*(n+)-t[].sum;
- }
- ll ans=;
- for(int i=;i<=mx;i++)ans+=1ll*(i-)*(h[i]-h[i-]);
- printf("%lld",ans);
- return ;
- }
【CodeForces】671 C. Ultimate Weirdness of an Array的更多相关文章
- 【CodeForces】671 D. Roads in Yusland
[题目]D. Roads in Yusland [题意]给定n个点的树,m条从下往上的链,每条链代价ci,求最少代价使得链覆盖所有边.n,m<=3*10^5,ci<=10^9,time=4 ...
- 【CodeForces】671 B. Robin Hood
[题目]B. Robin Hood [题意]给定n个数字的序列和k次操作,每次将序列中最大的数-1,然后将序列中最小的数+1,求最终序列极差.n<=5*10^5,0<=k<=10^9 ...
- 【Codeforces】Round #491 (Div. 2) 总结
[Codeforces]Round #491 (Div. 2) 总结 这次尴尬了,D题fst,E没有做出来.... 不过还好,rating只掉了30,总体来说比较不稳,下次加油 A:If at fir ...
- 【Codeforces】Round #488 (Div. 2) 总结
[Codeforces]Round #488 (Div. 2) 总结 比较僵硬的一场,还是手速不够,但是作为正式成为竞赛生的第一场比赛还是比较圆满的,起码没有FST,A掉ABCD,总排82,怒涨rat ...
- 【LeetCode】153. Find Minimum in Rotated Sorted Array 解题报告(Python)
[LeetCode]153. Find Minimum in Rotated Sorted Array 解题报告(Python) 标签: LeetCode 题目地址:https://leetcode. ...
- 【LeetCode】154. Find Minimum in Rotated Sorted Array II 解题报告(Python)
[LeetCode]154. Find Minimum in Rotated Sorted Array II 解题报告(Python) 标签: LeetCode 题目地址:https://leetco ...
- 【CodeForces】601 D. Acyclic Organic Compounds
[题目]D. Acyclic Organic Compounds [题意]给定一棵带点权树,每个点有一个字符,定义一个结点的字符串数为往下延伸能得到的不重复字符串数,求min(点权+字符串数),n&l ...
- 【Codeforces】849D. Rooter's Song
[算法]模拟 [题意]http://codeforces.com/contest/849/problem/D 给定n个点从x轴或y轴的位置p时间t出发,相遇后按对方路径走,问每个数字撞到墙的位置.(还 ...
- 【CodeForces】983 E. NN country 树上倍增+二维数点
[题目]E. NN country [题意]给定n个点的树和m条链,q次询问一条链(a,b)最少被多少条给定的链覆盖.\(n,m,q \leq 2*10^5\). [算法]树上倍增+二维数点(树状数组 ...
随机推荐
- alpha冲6
队名:日不落战队 安琪(队长) 今天完成的任务 回收站前端界面. 明天的计划 查看个人信息界面. 还剩下的任务 信息修改前端界面. 设置界面. 遇到的困难 模拟机莫名其妙就崩了,调试了很久,后在队友的 ...
- css声明的优先级
选择器的特殊性 选择器的特殊性由选择器本身的组件确定,特殊性值表述为4个部分,如0,0,0,0,0 一个选择器的具体特殊性如下确定 1.对于选择器给定的ID属性值,加0,1,0,0 2.对于选择器中给 ...
- 学生导师互选系统(php代码规范)
学生导师互选系统(php代码规范) php编码规范 组名:一不小心就火了 负责项目:学生导师互选系统(安卓端) 编写目的 为了更好的提高团队的的合作效率,保证开发的有效性和合理性,并可最大程度的提高程 ...
- 爬虫学习之-xpath
1.XPATH使用方法 使用XPATH有如下几种方法定位元素(相比CSS选择器,方法稍微多一点): a.通过绝对路径定位元素(不推荐!) WebElement ele = driver.findEle ...
- c语言基础笔记
一 :数据类型 1.float类型,在输出的时候可以使用 .数字 来把浮点数精确到小数点后几位,比如 printf("%.3f",float)精确到小数点后三位,不足补0 2.字 ...
- Spring MVC @RequestParam @RequestHeader @CookieValue用法
摘要: package com.hust.springmvc1; import org.springframework.stereotype.Controller; import org.spring ...
- 数据库引擎InnoDB和MyISAM区别
MyISAM是MySQL的默认数据库引擎(5.5版之前),由早期的ISAM(Indexed Sequential Access Method:有索引的顺序访问方法)所改良.虽然性能极佳,但却有一个缺点 ...
- UVA11735_Corner the Queens
题目是这样的,游戏规则,每个人轮流将二维空间上的皇后往下,往左或者往斜下45度的方向移动. 谁第一个移动到0,0的位置就获胜. 题目给定你若干个矩形,求矩形中任取一点且该点必胜的概率有概率. 其实是这 ...
- 迁移数据到历史表SQL(转)
有时工作需要需要把当前表的数据,移到历史表中,而历史表基本是以时间(年)为后缀来命名历史表的,如 A_2011,A_2012,在移数据时,要按数据的时间,移到不同的表中,且由于如果数据有同步.一次处理 ...
- Astronauts UVALive - 3713(2-SAT)
大白书例题 #include <iostream> #include <cstdio> #include <sstream> #include <cstrin ...