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 题解的更多相关文章

  1. HGOI 20181028 题解

    HGOI 20181028(复赛备考) /* 真是暴力的一天,最后一题MLE?由于数组开得太大了!!! 270滚粗 考场上好像智商高了很多?!(假的) */ sol:暴力求解,然后没有数据范围吐槽一下 ...

  2. HGOI 20190310 题解

    /* 又是又双叒叕WA的一天... 我太弱鸡了... 今天上午打了4道CF */ Problem 1 meaning 给出q组询问,求下列函数的值$ f(a) = \max\limits_{0 < ...

  3. HGOI 20190303 题解

    /* 记一串数字真难. 5435 今天比赛又是hjcAK的一天. 今天开题顺序是312,在搞T1之前搞了T3 昨天某谷月赛真是毒瘤. 但是讲评的同学不错,起码T4看懂了... 构造最优状态然后DP的思 ...

  4. HGOI 20180224 题解

    /* The Most Important Things: ljc chat with fyh on QQTa说期末考Ta数学74分感觉不好但是我觉得fyh是地表最强的鸭~~(of course en ...

  5. HGOI 20190218 题解

    /* 又是AK局... hjc又双叒叕AK了... Hmmm...我侥幸 */ Problem A card 给出无序序列a[]可以选择一个数插入到合适的位置作为一次操作,至少多少次操作后可以把序列变 ...

  6. HGOI 20190217 题解

    /* for me,开训第一天 /beacuse 文化课太差被抓去补文化课了... 看一眼题 : AK局? 但是,Wa on test #10 in problem C 290! (就差那么一咪咪) ...

  7. NOI.AC 20181103 题解

    CF 1037B  Reach Median 班上 n个同学(n 是奇数)排成一排站队,为了美观,需要大家高度的中位数是 x. 你可以让同学们在脚下垫木板或者稍微蹲一点来达成这个目标.对任意一位同学的 ...

  8. HGOI 20181101题解

    /* 又是爆0的一天(不知道今年高考难不难,反正今天(信息学)真的难!) */ solution:对于两个数相加,有一个显然的结论就是要么不进位(相对于位数大的),要么(进最多一位) 然后对于整个数组 ...

  9. HGOI 20191108 题解

    Problem A 新婚快乐 一条路,被$n$个红绿灯划分成$n+1$段,从前到后一次给出每一段的长度$l_i$,每走$1$的长度需要$1$分钟. 一开始所有红绿灯都是绿色的,$g$分钟后所有红绿灯变 ...

随机推荐

  1. MySQL学习笔记04 插入中文时出现ERROR 1366 (HY000)

    1 环境: MySQL Server 6.0  命令行工具 2 问题 :  插入中文字符数据出现如下错误: ERROR 1366 (HY000): Incorrect string value: '\ ...

  2. 20155318 《网络攻防》 Exp8 Web基础

    20155318 <网络攻防> Exp8 Web基础 基础问题 什么是表单? HTML表单用于收集用户输入,用元素定义,包含不同类型的input元素.复选框.单选按钮.提交按钮等等.一个表 ...

  3. 20155330 《网络对抗》 Exp8 Web基础

    20155330 <网络对抗> Exp8 Web基础 实验问题回答 什么是表单 表单可以收集用户的信息和反馈意见,是网站管理者与浏览者之间沟通的桥梁. 一个表单有三个基本组成部分 表单标签 ...

  4. 滚动条ScrollViewer防止滚动时按内容跳跃式滚动的设置

    原文:滚动条ScrollViewer防止滚动时按内容跳跃式滚动的设置 属性中将CanContentScroll设置为False,滚动时就不会跳了,会连续的滚动

  5. Android开发——JobScheduler机制

    年Google开发大会上指出,如果每个APP都使用这个API,那么可以节约15%到20%的电量. 2.  JobScheduler拥有更多的触发选项 JobScheduler比AlarmManager ...

  6. Kubernetes学习之路(二十一)之网络模型和网络策略

    目录 Kubernetes的网络模型和网络策略 1.Kubernetes网络模型和CNI插件 1.1.Docker网络模型 1.2.Kubernetes网络模型 1.3.Flannel网络插件 1.4 ...

  7. 【第六课】Nginx常用配置下详解

    目录 Nginx常用配置下详解 1.Nginx虚拟主机 2.部署wordpress开源博客 3.部署discuz开源论坛 4.域名重定向 5.Nginx用户认证 6.Nginx访问日志配置 7.Ngi ...

  8. 蓝牙inquiry流程之HCI_Inquiry_Result_With_RSSI和HCI Extended Inquiry Result处理

    首先介绍一下和inquiry的相关的流程. inquiry是从协议栈下发的一个HCI命令.其格式如下: 这里简单介绍下第二个参数,是inquiry的持续时间, 从上图看出 inquiry持续的时间是 ...

  9. eclipse 最最最常用快捷键

    使用eclipse这么久,发现其跟PS一样,使用一些快捷键会有效率很多. 至此总结出以下每次打开eclipse基本都会用上的快捷键. 不熟悉这些快捷键,在实际编程中有意识使用的话对以后编码很有帮助. ...

  10. css小技巧::not()选择器的妙用

    比如,要实现下面的效果(例如:一个列表的最后一项没有边框): See the Pen gmrGOV by 杨友存 (@Gavin-YYC) on CodePen. 一般的文档结构如下: <!-- ...