Gym102832K. Ragdoll(CCPC长春)

Gym102832K. Ragdoll(CCPC长春)

题意:

n个点,每个点都有自己的权值aia_iai,一开始第i个点在第i个集合里。
如果一对点(i,j)是bad当且仅当满足i和j在同一个集合里,且gcd(ai,aj)=ai⊕ajgcd(a_i,a_j)=a_i⊕a_jgcd(ai,aj)=aiaj
现在有3种操作:

  1. 新增一个点x(在新的一个集合里),值为v
  2. 合并两个集合x和y
  3. 将点x的值改为v

有m个操作,请输出每次操作后有多少对点是bad

题解:

一开始无从下手,因为这个式子不知道该如何应用
这里有个很巧妙的转换,ai⊕aj​⩾∣ai−aj∣​⩾gcd(ai,aj)a_i⊕a_j ​⩾|a_i-a_j|​⩾gcd(a_i,a_j)aiajaiajgcd(ai,aj),如果我们要让等式成立,则需要满足:

  1. ai​⊕aj=∣ai−aj∣a_i​⊕a_j=|a_i-a_j|aiaj=aiaj
  2. ∣ai−aj∣=gcd(ai,aj)|a_i-a_j|=gcd(a_i,a_j)aiaj=gcd(ai,aj)

转换证明:
a⊕b>=a−ba⊕b>=a-bab>=ab,这个式子是显然成立的,用二进制思想去考虑
gcd(a,b)<=a−bgcd(a,b)<=a-bgcd(a,b)<=ab:由于gcd(a,b)是b的因子,gcd(a,b)是a的因子,所以gcd(a,b)就是(a-b)的因子,所以gcd(a,b)一定小于等于a-b

∣ai−aj∣=gcd(ai,aj)|a_i-a_j|=gcd(a_i,a_j)aiaj=gcd(ai,aj)
aj=ai+gcd(ai,aj)a_j=a_i+gcd(a_i,a_j)aj=ai+gcd(ai,aj)或者aj=ai−gcd(ai,aj)a_j=a_i-gcd(a_i,a_j)aj=aigcd(ai,aj)
gcd(ai,aj)gcd(a_i,a_j)gcd(ai,aj)aia_iai的因子
因此我们可以枚举aia_iai,然后枚举aia_iai的因子,这样就求出对于aia_iai而言合法的aja_jaj,用vector存下来。相当于我们可以预处理出所有是bad的点对
我们再看三个操作,这种集合操作,都可以用并查集来维护
对于操作1,就是新加一个集合
对于操作2,合并两个集合,如果我们直接盲目合并,复杂度是O(n),这里用到启发式合并,把小的集合合并到大的集合中,这样复杂度降低为O(log)
对于操作3,我们先去掉原先元素可能组成的bad点对,改成数v后再加入新的bad点对
在每次操作中,维护着答案,如何维护呢?因为我们预处理出所有点对,合并时两个集合时,就枚举小集合所有元素,看大集合内有多少可以构成bad点对,更新答案ans。修改数时就是删除原先,加上新填
unordered_map速度更快
详细内容看代码

代码:

#include <bits/stdc++.h>
#include <unordered_map>
#define debug(a, b) printf("%s = %d\n", a, b);
using namespace std;
typedef long long ll;
typedef unsigned long long ull;
typedef pair<int, int> PII;
clock_t startTime, endTime;
//Fe~Jozky
const ll INF_ll= 1e18;
const int INF_int= 0x3f3f3f3f;
void read(){};
template <typename _Tp, typename... _Tps> void read(_Tp& x, _Tps&... Ar)
{x= 0;char c= getchar();bool flag= 0;while (c < '0' || c > '9')flag|= (c == '-'), c= getchar();while (c >= '0' && c <= '9')x= (x << 3) + (x << 1) + (c ^ 48), c= getchar();if (flag)x= -x;read(Ar...);
}
template <typename T> inline void write(T x)
{if (x < 0) {x= ~(x - 1);putchar('-');}if (x > 9)write(x / 10);putchar(x % 10 + '0');
}
void rd_test()
{
#ifdef ONLINE_JUDGE
#elsestartTime = clock ();freopen("data.in", "r", stdin);
#endif
}
void Time_test()
{
#ifdef ONLINE_JUDGE
#elseendTime= clock();printf("\nRun Time:%lfs\n", (double)(endTime - startTime) / CLOCKS_PER_SEC);
#endif
}
const int maxn=5e5+9;
int n,m; 
int a[maxn];
vector<int>g[maxn];
int fa[maxn];
int sz[maxn];
ll ans=0;
unordered_map<int,ll>mp[maxn];
int gcd(int a,int b){if(b)return gcd(b,a%b);return a;
}
int find(int x){if(fa[x]==x)return x;return fa[x]=find(fa[x]);
} 
void work(int x){for(int i=1;i*i<=x;i++){//枚举因子 if(x%i==0){int tmp=x-i;if((tmp^x)==gcd(tmp,x)&&tmp!=0)//符合bad的点对 g[x].push_back(tmp);tmp=x+i; if((tmp^x)==gcd(tmp,x)&&tmp<=200000)g[x].push_back(tmp);if(i!=x/i){tmp=x-x/i;if((tmp^x)==gcd(tmp,x)&&tmp!=0)g[x].push_back(tmp);tmp=x+x/i;if((tmp^x)==gcd(tmp,x)&&tmp!=200000)g[x].push_back(tmp);} }}
}
void merge(int x,int y){int fx=find(x);int fy=find(y);if(fx==fy)//如果在一个集合退出 return ;//我们让fy是小集合,用fy去合并到fx if(sz[fx]<sz[fy]) swap(fx,fy);//合并时更新答案ans for(auto& it:mp[fy]){//遍历小集合内的所有元素it.first
//		cout<<"it.first="<<it.first<<endl;for(int i=0;i<g[it.first].size();i++){
//			printf("g[it.first].size()=%d\n",g[it.first].size());
//			printf("g[it.first][%d]=%d\n",i,g[it.first][i]);//看大集合中有多少点可以与小集合的元素组成bad点对 if(mp[fx].count(g[it.first][i]))ans+= 1ll * it.second * mp[fx][g[it.first][i]];}}for(auto &it:mp[fy]){//把fy的所有元素存放到集合fx中,即合并 mp[fx][it.first]+=it.second;} mp[fy].clear();sz[fx]+=sz[fy];fa[fy]=fx;
}
int main()
{rd_test();for(int i=1;i<=200002;i++){work(i);//预处理出所有bad点对 }cin>>n>>m;for(int i=1;i<=n+m;i++){fa[i]=i; sz[i]=1;//集合大小}for(int i=1;i<=n;i++){cin>>a[i];mp[i][a[i]]++;//每个集合元素以及大小 }for(int i=1;i<=m;i++){//添加 int op,x,y;cin>>op>>x>>y;if(op==1){a[x]=y;sz[x]=1;mp[x][y]=1;}else if(op==2){//合并操作 merge(x,y);}else if(op==3){//修改值 int u=find(x);for(int i=0;i<g[a[x]].size();i++){//删除原集合内的bad点对 if(mp[u].count(g[a[x]][i]))ans-=mp[u][g[a[x]][i]]; }mp[u][a[x]]--;//点a[x]删除a[x]=y;//修改新值for(int i=0;i<g[a[x]].size();i++){if(mp[u].count(g[a[x]][i]))ans+=mp[u][g[a[x]][i]]; }mp[u][a[x]]++;//加上新点 }printf("%lld\n",ans); }return 0;//Time_test();
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/315914.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

LCA模板

LCA模板 此处只想贴个代码。 void dfs(int x,int father,int Dep) {dep[x]Dep;for (int i1;i<Log[Dep-1];i){Ln[x][i]Ln[ Ln[x][i-1] ][i-1];Ls[x][i]Ln[ Ln[x][i-1] ][i-1];}for (int i0;i<e[x].size;i)if (e[x][i].to!father){Ln[ e[x][i].to ][0]x;Ls[ e[x][i].to ]…

从阿里中台战略看企业IT架构转型之道(下)

此文是我阅读《企业IT架构转型之道》一书的学习笔记的下半部分&#xff0c;所有内容出自钟华老师的这本书。上半部分Part1~Part5请点击这里Part 6 异步与缓存原则异步化事务 > 核心是ACID柔性事务 > 基础是CAP理论和BASE理论&#xff0c;因为互联网应用最核心的需求是高可…

CF1543C. Need for Pink Slips

CF1543C. Need for Pink Slips 题意&#xff1a; 题解&#xff1a; 其实具体的计算方法在说明里面都写了&#xff1a;对于第一个数据&#xff1a; 0.2 0.2 0.6 0.2组成方案如下&#xff1a; 就是c和m如果大于v就减&#xff0c;小于v就变成0&#xff0c;到p直接停止 所以直接…

并查集(并茶几)

并查集&#xff08;并茶几&#xff09;的应用 一、What‘s that&#xff1f; 并查集是一种树型的数据结构&#xff0c;用于处理一些不相交集合&#xff08;Disjoint Sets&#xff09;的合并及查询问题。常常在使用中以森林来表示。 ——百度百科 二、How to uphold 0.我们的需…

从阿里中台战略看企业IT架构转型之道(上)

此文是我阅读《企业IT架构转型之道》一书的学习笔记的上半部分&#xff0c;所有内容出自钟华老师的这本书。零、为何阅读《企业IT架构转型之道》在加入X公司后&#xff0c;开始了微服务架构的实践&#xff0c;也开始了共享平台服务的建设&#xff0c;在这方面阿里巴巴的中台战略…

ML.NET机器学习、API容器化与Azure DevOps实践(四):持续集成与k8s持续部署

通过上文所介绍的内容&#xff0c;我们已经完成了RESTful API的开发&#xff0c;现在&#xff0c;就可以使用Azure DevOps来进行持续集成&#xff08;CI&#xff09;和k8s持续部署&#xff08;CD&#xff09;了。本文我会对使用Azure DevOps进行CI/CD的过程中需要注意的地方进行…

P3195 [HNOI2008]玩具装箱

P3195 [HNOI2008]玩具装箱 题意&#xff1a; n件玩具&#xff0c;第i件玩具经过压缩后的一维长度为CiC_iCi​,现在把玩具装入一维容器中&#xff0c;要求&#xff1a; 在一个一维容器中的玩具编号是连续的如果一个一维容器中有多个玩具&#xff0c;那么两件玩具之间要加入一…

卷积与莫比乌斯反演

卷积与莫比乌斯反演 目录 卷积与莫比乌斯反演 0前言 0.1前置技能 0.2问题的引入 1.简单定义 1.1数论函数的定义 1.2卷积的定义 1.3反演的基本形式 2.1莫比乌斯反演 3.1例题&#xff1a;【luogu-P2257 YY的GCD】 题目大意&#xff1a; solution1 solution2 0.前言 莫比…

ML.NET机器学习、API容器化与Azure DevOps实践(三):RESTful API

通过上文所述案例&#xff0c;我们已经选择了最优回归算法来预测学生的综合成绩&#xff0c;并且完成了基于训练数据集的预测模型训练。从实现上&#xff0c;训练好的模型被保存成一个ZIP文件&#xff0c;以便在其它项目中直接调用以完成机器学习的实践场景。在本文中&#xff…

Manthan, Codefest 19 (open for everyone, rated, Div. 1 + Div. 2)

Manthan, Codefest 19 (open for everyone, rated, Div. 1 Div. 2) 题号题目知识点AXORinacciBUniqueness尺取CMagic Grid构造DRestore Permutation思维线段树ELet Them Slide线段树思维FBits And PiecesGPolygonsHRed Blue Tree

杜教筛

杜教筛 1.概述 杜教筛是用以解决积性函数前缀和的算法。 在学习了莫比乌斯反演之后&#xff0c;杜教筛的过程就会显得简单而自然。 2.基本形式 对于积性函数&#xff0c;我们定义如下函数&#xff1a; 构造积性函数 &#xff0c;使得 显然 &#xff1a; 进一步转化&#xf…

CF1208D Restore Permutation

CF1208D Restore Permutation 题意&#xff1a; 现在有一个从1到n的一个全排列,但是你不知道这个排列到底是什么,但是你有一个sum[i],其中sum[i]表示∑j1i−1(aj<ai)?aj:0∑_{j1}^{i−1}(a_j<a_i)?a_j:0∑j1i−1​(aj​<ai​)?aj​:0,现在给你sum数组,让你求出这…

ML.NET机器学习、API容器化与Azure DevOps实践(二):案例

在上文中&#xff0c;我简单地介绍了机器学习以及ML.NET的相关知识&#xff0c;从本讲开始&#xff0c;我会基于一个简单的案例&#xff1a;学生成绩预测&#xff0c;来介绍使用ML.NET进行机器学习以及API部署的基本过程。本案例的数据来源为加州大学尔湾分校的机器学习公开样本…

业界萌新对斯坦纳树的小结

业界萌新对斯坦纳树的小结 0.简介 斯坦纳树问题是组合优化问题&#xff0c;与最小生成树相似&#xff0c;是最短网络的一种。最小生成树是在给定的点集和边中寻求最短网络使所有点连通。而最小斯坦纳树允许在给定点外增加额外的点&#xff0c;使生成的最短网络开销最小。 ——…

cf1208E. Let Them Slide

cf1208E. Let Them Slide 题意&#xff1a; 都放在一个长度为W的框里面。有n个序到&#xff0c;第i个序列的长度是1。这些序到并排放在一起&#xff0c;每一个序列都放在一个长度为w的框里 这些序列可以在框里面滑动&#xff0c;但是不能划出框。 对于每一个位置&#xff0…

结合eShopOnWeb全面认识领域模型架构

一.项目分析在上篇中介绍了什么是"干净架构"&#xff0c;DDD符合了这种干净架构的特点&#xff0c;重点描述了DDD架构遵循的依赖倒置原则&#xff0c;使软件达到了低藕合。eShopOnWeb项目是学习DDD领域模型架构的一个很好案例&#xff0c;本篇继续分析该项目各层的职…

SOS_dp算法

Codeforces博客 简介&#xff1a; 前置知识&#xff1a;状压dp Sum over Subsets dynamic programming&#xff0c;简称Sos dp,状压dp的一种 用一个列题引出SOS dp&#xff1a; 给你一个由2N2^N2N个整数组成的确定数组A&#xff0c;我们需要计算对于任意的x&#xff0c;F(x)所…

微软开源Bing搜索背后的关键算法

微软今天宣布开源了一项 Bing 搜索背后的关键算法 —— SPTAG&#xff0c;它使 Bing 能够快速将搜索结果返回给用户。仅在几年前&#xff0c;网络搜索很简单&#xff0c;用户输入几个关键词然后浏览结果页面。现如今&#xff0c;这些用户可能会在手机上拍照并将其放入搜索框中&…

Stern-Brocot Tree

Stern-Brocot Tree 0.简介 Stern-Brocot Tree&#xff0c;俗称SB树&#xff08;滑稽&#xff09;。它能够表示出所有的最简分数&#xff0c;如下图。 1.一些规律 显然&#xff0c;对于两个相邻的最简分数 可以得到另一个最简分数 这样就可以在Stern-Brocot Tree上表示出所有…

FWT(快速沃尔什变换)

文章目录引入&#xff1a;or卷积and卷积xor卷积IFWT模板&#xff1a;例题&#xff1a;引入&#xff1a; FFT/NTT是用来解决∑ijkA[i]B[j]\sum_{ijk}A[i]B[j]∑ijk​A[i]B[j]的式子 而FWT是用来解决Ci∑j⊕kiAjBkC_i\sum_{j⊕ki}A_jB_kCi​∑j⊕ki​Aj​Bk​ ​ FWT是一种用于处…