CF510E. Fox And Dinner

https://codeforces.com/contest/510

分析:

  • 由于\(a_i>2\), 相邻两个数一定一奇一偶,按奇偶建立二分图。
  • 环上每个点度数都为2,因此只需要找是否每个点都能匹配两个。
  • 建图跑dinic即可。

代码:

#include <cstdio>
#include <cstring>
#include <cstdlib>
#include <algorithm>
#include <cmath>
using namespace std;
#define N 1050
#define M 2000050
#define S 1049
#define T 1048
#define inf 0x3f3f3f3f
int head[M],nxt[M],to[M],cnt=1,n,flow[M];
int dep[N],Q[N],a[N];
int vis[M],pri[M],pri_cnt,ch[N][2];
int vec[N][N],vsize[N],inv[N];
void sieve() {
int i,j,lim=15000;
for(i=2;i<=lim;i++) {
if(!vis[i]) {
pri[++pri_cnt]=i;
}
for(j=1;j<=pri_cnt&&i*pri[j]<=lim;j++) {
vis[i*pri[j]]=1;
if(i%pri[j]==0) break;
}
}
}
inline void add(int u,int v,int f) {
to[++cnt]=v; nxt[cnt]=head[u]; head[u]=cnt; flow[cnt]=f;
to[++cnt]=u; nxt[cnt]=head[v]; head[v]=cnt; flow[cnt]=0;
}
bool bfs() {
int i;
memset(dep,0,sizeof(dep));
int l=0,r=0;
Q[r++]=S; dep[S]=1;
while(l<r) {
int x=Q[l++];
for(i=head[x];i;i=nxt[i]) if(flow[i]&&!dep[to[i]]) {
dep[to[i]]=dep[x]+1;
if(to[i]==T) return 1;
Q[r++]=to[i];
}
}
return 0;
}
int dfs(int x,int mf) {
int i,nf=0;
if(x==T) return mf;
for(i=head[x];i;i=nxt[i]) if(flow[i]&&dep[to[i]]==dep[x]+1) {
int tmp=dfs(to[i],min(mf-nf,flow[i]));
if(!tmp) dep[to[i]]=0;
nf+=tmp;
flow[i]-=tmp;
flow[i^1]+=tmp;
if(nf==mf) break;
}
return nf;
}
void dinic() {
while(bfs()) {
while(dfs(S,inf)>0);
}
}
bool check(int x) {
int i;
for(i=1;i<=pri_cnt&&pri[i]*pri[i]<=x;i++) if(x%pri[i]==0) return 0;
return 1;
}
int main() {
sieve();
scanf("%d",&n);
int i,j,x;
for(i=1;i<=n;i++) scanf("%d",&a[i]);
for(i=1;i<=n;i++) {
if(!(a[i]&1)) {
add(S,i,2);
for(j=1;j<=n;j++) if(i!=j&&(a[j]&1)) { if(check(a[i]+a[j])) {
add(i,j,1);
if(i==1&&j==3) puts("FUCK");
}
}
}else {
add(i,T,2);
}
}
dinic();
int ERR=0;
for(x=1;x<=n;x++) {
if(!(a[x]&1)) {
for(i=head[x];i;i=nxt[i]) if(to[i]!=S) {
if(flow[i]==0) {
if(!ch[x][0]) ch[x][0]=to[i];
else ch[x][1]=to[i];
}
}
if(!ch[x][1]) {ERR=1; break;}
}else {
for(i=head[x];i;i=nxt[i]) if(to[i]!=T) {
if(flow[i]==1) {
if(!ch[x][0]) ch[x][0]=to[i];
else ch[x][1]=to[i];
}
}
if(!ch[x][1]) {ERR=1; break;}
}
}
if(ERR) {puts("Impossible"); return 0;}
memset(vis,0,sizeof(vis));
int tot=0;
for(i=1;i<=n;i++) if(!vis[i]) {
int tp=0;
memset(inv,0,sizeof(inv));
vec[++tot][++tp]=i; inv[i]=1;
for(j=1;;j++) {
int p=vec[tot][j];
inv[p]=1;
if(inv[ch[p][0]]&&inv[ch[p][1]]) break;
if(inv[ch[p][0]]) {
vec[tot][++tp]=ch[p][1];
}else {
vec[tot][++tp]=ch[p][0];
}
}
for(j=1;j<=tp;j++) vis[vec[tot][j]]=1;
vsize[tot]=tp;
}
printf("%d\n",tot);
for(i=1;i<=tot;i++) {
printf("%d ",vsize[i]);
for(j=1;j<=vsize[i];j++) {
printf("%d ",vec[i][j]);
}
puts("");
}
}

最新文章

  1. SQL SERVER 临时表导致存储过程重编译(recompile)的一些探讨
  2. Unity3D 查找Update函数体为空的类
  3. spring+IOC+DI+AOP优点分析(一)
  4. Enterprise Solution 2.3
  5. gogo
  6. hibernate_validator_02
  7. AES - Rijndael 算法(三)
  8. 支付宝移动支付开发详细教程服务端采用.net mvc webapi(C#)
  9. 在window上安装redis
  10. C语言-第2次作业得分
  11. Python之路(第三十七篇)并发编程:进程、multiprocess模块、创建进程方式、join()、守护进程
  12. Hdoj 2108.Shape of HDU 题解
  13. Java 集合总体框架介绍
  14. WPF listbox的分组研究
  15. UVa LA 3695 - Distant Galaxy 前缀和,状态拆分,动态规划 难度: 2
  16. snort帮助文档
  17. SQL Server T—SQL 语句【建 增 删 改】(建外键)
  18. solr的随机排序 【转载】
  19. Linux下lshw,lsscsi,lscpu,lsusb,lsblk硬件查看命令
  20. &lt;2014 05 16&gt; 线性表、栈与队列——一个环形队列的C语言实现

热门文章

  1. Windows系统SVN服务器搭建与使用
  2. json:js和jquery中轻量级数据交换格式
  3. js的new到底干了啥 -
  4. TFS 中工作项的定制-修改工作流
  5. BZOJ3211花神游历各国
  6. python数据分析之:时间序列一
  7. centos7 运行postgres 数据库脚本db.sql
  8. Spring注解式与配置文件式
  9. 基于Spring框架的Shiro配置(转发:http://kdboy.iteye.com/blog/1103794)
  10. MD5文件