HGOI 20181103 题解

problem:把一个可重集分成两个互异的不为空集合,两个集合里面的数相乘的gcd为1(将集合中所有元素的质因数没有交集)
solution:显然本题并不是那么容易啊!考场上想了好久。。
其实转化为上面的题意就简单多了,对于每一个元素分解质因数,就是在筛质数的时候记下每一个合数的最小质因子low[x],然后每一次不停的除low[x]
得到一个合数x',然后继续除low[x']即可,然后我们统计出含有每一个质因子的数有哪些(用一个二维vector),然后具有同种质因子的数必须放在同一个集合里面,
也就是说他们可以合并在一起,考虑并查集处理,就把每一个质因子把对应的数全部合并在一起,然后最后统计剩下的没有交集的最终不能再合并的互异块的个数tot
考虑用tot分成两个不同的非空集合,显然是2 tot -2,快速幂处理即可。
复杂度O(n log n)
code:
# include <bits/stdc++.h>
# define int long long
# define pow Pow
using namespace std;
const int M=1e6+;
const int mo=1e9+;
bool prime[M];
int low[M],a[M],f[M],n;
vector<int>r[M];
inline int read()
{
int X=,w=; char c=;
while (!(c>=''&&c<='')) w|=c=='-',c=getchar();
while (c>=''&&c<='') X=(X<<)+(X<<)+(c^),c=getchar();
return w?-X:X;
}
void getprime(int limit)
{
memset(low,,sizeof(low));
memset(prime,true,sizeof(prime));
for (int i=;i<=limit;i++) {
if (!prime[i]) continue;
for (int j=i+i;j<=limit;j+=i) {
if (low[j]==) low[j]=i;
prime[j]=false;
}
}
}
int father(int x)
{
if (f[x]==x) return x;
f[x]=father(f[x]);
return f[x];
}
int calc(int x,int y)
{
int fx=father(x),fy=father(y);
f[fx]=fy;
}
void solve(int id)
{
int num=a[id];
while (low[num]) {
r[low[num]].push_back(id);
int tmp=low[num];
while (num%tmp==) num/=tmp;
}
r[num].push_back(id);
}
int pow(int x,int n)
{
int ans=;
while (n) {
if (n&) ans=ans*x%mo;
x=x*x%mo;
n>>=;
}
return ans%mo;
}
signed main()
{
int T=read();
while (T--) {
n=read();
int MAX=;
for (int i=;i<=n;i++) {
a[i]=read(); f[i]=i;
MAX=max(MAX,a[i]);
}
getprime(MAX);
for (int i=;i<=M;i++) r[i].clear();
for (int i=;i<=n;i++) solve(i);
for (int i=;i<=M;i++) {
if (r[i].size()==) continue;
int k=r[i][];
for (int j=;j<r[i].size();j++) calc(k,r[i][j]);
}
int ret=;
for (int i=;i<=n;i++) if (f[i]==i) ret++;
printf("%lld\n",(pow(,ret)%mo-+mo)%mo);
}
return ;
}


sol:分成两组然后每一组的人都互相认识,显然想到对每一个联通块01染色分成不同的集合
对于每一个联通块统计出0的块的个数1的块的个数,显然0或者1的个数为n/2最好才会使 calc(i)+calc(n-i)最小
其中calc(x)=x(x-1)/2,
由于每一个块我们只能有1或者0,那么设f[i][j]表示前i个块,选择0或者1的块为j是否可能
转移的话就是
f[i][j]|=f[i-1][j-a[i].cnt0]|f[i-1][j-a[i].cnt1]
然后j从n/2向左枚举然后找到若f[n][j]合法
那么最小化 clac(j)+calc(n-j)即可
code:
# include <bits/stdc++.h>
# define int long long
using namespace std;
const int MAXN=;
int mp[MAXN][MAXN],n,m,col[MAXN];
bool f[MAXN][MAXN];
int cnt0,cnt1;
int o;
struct rec{ int cnt0,cnt1;}b[MAXN];
void dfs(int u,int fa,int c)
{
col[u]=c;
//printf("%d ; col=%d\n",u,col[u]);
if (c==) cnt0++; else cnt1++;
for (int v=;v<=n;v++) {
if (mp[u][v]==) continue;
if (col[v]!=-) {
if (col[v]!=c^) { puts("-1"); exit();}
else continue;
}
dfs(v,u,c^);
}
}
int calc(int x){ return x*(x-)/;}
signed main()
{
scanf("%lld%lld",&n,&m);
for (int i=;i<=n;i++)
for (int j=;j<=n;j++)
if (i==j) mp[i][j]=false;
else mp[i][j]=true;
while (m--) {
int u,v; scanf("%lld%lld",&u,&v);
mp[u][v]=mp[v][u]=false;
}
memset(col,-,sizeof(col));
for (int i=;i<=n;i++)
if (col[i]==-) {
cnt0=cnt1=;
dfs(i,-,);
b[++o].cnt1=cnt1;
b[++o].cnt0=cnt0;
}
//f[i][j]前i个集合,到达A中有j个是否成立
//f[i][j]|=f[i-1][j-b[i].cnt0]|f[i-1][j-b[i].cnt1]
memset(f,false,sizeof(f));
f[][b[].cnt1]=f[][b[].cnt0]=true;
for (int i=;i<=n;i++)
for (int j=;j<=n;j++)
f[i][j]=f[i][j]|f[i-][j-b[i].cnt0]|f[i-][j-b[i].cnt1];
int Ans=n*n*;
for (int i=;i<=n;i++)
if (f[n][i]) Ans=min(Ans,calc(i)+calc(n-i));
cout<<Ans<<endl;
return ;
}
HGOI 20181103 题解的更多相关文章
- HGOI 20181028 题解
HGOI 20181028(复赛备考) /* 真是暴力的一天,最后一题MLE?由于数组开得太大了!!! 270滚粗 考场上好像智商高了很多?!(假的) */ sol:暴力求解,然后没有数据范围吐槽一下 ...
- HGOI 20190310 题解
/* 又是又双叒叕WA的一天... 我太弱鸡了... 今天上午打了4道CF */ Problem 1 meaning 给出q组询问,求下列函数的值$ f(a) = \max\limits_{0 < ...
- HGOI 20190303 题解
/* 记一串数字真难. 5435 今天比赛又是hjcAK的一天. 今天开题顺序是312,在搞T1之前搞了T3 昨天某谷月赛真是毒瘤. 但是讲评的同学不错,起码T4看懂了... 构造最优状态然后DP的思 ...
- HGOI 20180224 题解
/* The Most Important Things: ljc chat with fyh on QQTa说期末考Ta数学74分感觉不好但是我觉得fyh是地表最强的鸭~~(of course en ...
- HGOI 20190218 题解
/* 又是AK局... hjc又双叒叕AK了... Hmmm...我侥幸 */ Problem A card 给出无序序列a[]可以选择一个数插入到合适的位置作为一次操作,至少多少次操作后可以把序列变 ...
- HGOI 20190217 题解
/* for me,开训第一天 /beacuse 文化课太差被抓去补文化课了... 看一眼题 : AK局? 但是,Wa on test #10 in problem C 290! (就差那么一咪咪) ...
- NOI.AC 20181103 题解
CF 1037B Reach Median 班上 n个同学(n 是奇数)排成一排站队,为了美观,需要大家高度的中位数是 x. 你可以让同学们在脚下垫木板或者稍微蹲一点来达成这个目标.对任意一位同学的 ...
- HGOI 20181101题解
/* 又是爆0的一天(不知道今年高考难不难,反正今天(信息学)真的难!) */ solution:对于两个数相加,有一个显然的结论就是要么不进位(相对于位数大的),要么(进最多一位) 然后对于整个数组 ...
- HGOI 20191108 题解
Problem A 新婚快乐 一条路,被$n$个红绿灯划分成$n+1$段,从前到后一次给出每一段的长度$l_i$,每走$1$的长度需要$1$分钟. 一开始所有红绿灯都是绿色的,$g$分钟后所有红绿灯变 ...
随机推荐
- OutputFormat输出过程的学习
花了大约1周的时间,最终把MapReduce的5大阶段的源代码学习结束掉了.收获不少.就算本人对Hadoop学习的一个里程碑式的纪念吧.今天花了一点点的时间,把MapReduce的最后一个阶段.输出O ...
- 【H5】直接拨打电话
一般<a href="tel:400-663-5999">400-663-5999</a>实现. 而形如<a href="tel:*9204 ...
- [Baltic2013]ballmachine BZOJ3133
分析: 我们考虑,因为每次放置的时候,都是向子树中含有的编号最小的哪一个走,那么放置的顺序是固定的,我们将边以to的子树最小排序,之后得到的出栈序就是球的放入顺序.目测可以使用堆来实现,线段树也能实现 ...
- C# online update demo
using System; using System.Collections.Generic; using System.Linq; using System.Text; using System.W ...
- 网络对抗技术 2017-2018-2 20155215 Exp9 Web安全基础
1.实践过程 前期准备:WebGoat WebGoat分为简单版和开发板,简单版是个Java的Jar包,只需要有Java环境即可,我们在命令行里执行java -jar webgoat-containe ...
- 20155333 《网络对抗》 Exp7 网络欺诈防范
20155333 <网络对抗> Exp7 网络欺诈防范 基础问题 通常在什么场景下容易受到DNS spoof攻击? 公共网络 在日常生活工作中如何防范以上两种攻击方法? DNS欺骗攻击是很 ...
- 记一次SpringMVC碰到的坑
在SpringMVC中,我们Controller中接收比如表单的参数,只要保证方法的形参的名字和表单中input元素的的name一样就可以接收到参数. 但是,我开发的一 ...
- Struts2将图片输出到页面
在做CRUD的过程中,添加页面是个表单,表单里面有一项是上传头像文件.这样表单提交后,头像文件上传了. 但这个文件存的地址是本地硬盘的一个文件夹.在编辑页面要做这个头像的回显的话,就需 ...
- 5.Xilinx RapidIO核例子工程源码分析
https://www.cnblogs.com/liujinggang/p/10091216.html 一.软件平台与硬件平台 软件平台: 操作系统:Windows 8.1 64-bit 开发套件:V ...
- PowerBI开发 第十一篇:报表设计技巧(更新)
PowerBI版本在持续的更新,这使得报表设计能够实现更多新的功能,您可以访问 PowerBI Blog查看PowerBI的最新更新信息,本文总结了PowerBI新版本的重要更新和设计技巧. 我的Po ...