【解析】欧拉筛法,奇偶分析。建二分图,网络流



[Analysis]



http://blog.csdn.net/qq574857122/article/details/43453087。



所谓的连通块就是指满流了,因为我直接使用剩余流量求网络流。

至于怎样推断满流,我想到下面几种:

①对于初始的网络。若图的流向是一样的。那么就直接推断对于一条边k的正向边k>>1<<1是否满流。

②对于一般的,能够存流量限制。

③或者对每条边再存个tag。若tag=1,则表示初始时这条边容量限制非0,否则是0。



[Q&A]

问题1:为什么这样搜索能够出解而不错误?

回答1:这不是非常明显的嘛,因为满流了,所以除原点和汇点外的每一个节点连接且仅连接两个节点。

这样从一个节点过来。那么必定仅仅能从还有一个节点出去。

最后必定会有一个节点连接到第一个节点。这是就停止了。假如没有。那么一直找到了第n个节点就找不到了。矛盾。

特殊的,对于每一个连通集合的第一个节点,选择了随意一个相邻的节点,这也是没问题的。



[Sum]

①对于素数的问题,要想到奇偶分析。

②k!=-1。等价于~k,这里能够化简。事实上scanf("%d",&a)这个函数假设读不到不论什么东西,返回的值也是-1。

之所以能这样由于-1的二进制为最大即2^k-1,取反后为0。而其它取反都非0。

③推断满流的三种办法。

④回想了网络流算法。

[Code]

#include <cstdio>
#include <cstring>
#include <cstdlib>
using namespace std; const int N=240;
const int M=N*N;
const int P=20001; int n,odd[N],even[N];
int w[N],pri[P],vp[P];
struct G
{
int v,f,nxt;
}map[M+M];
int tt,hd[N];
int level[N],q[N],h,t;
G list[N]; int tl,used[N],fs[N],num,cnt[N]; inline int read(void)
{
int s=0; char c=getchar();
for (;c<'0'||c>'9';c=getchar());
for (;'0'<=c&&c<='9';c=getchar()) s=s*10+c-'0';
return s;
} inline void ins(int u,int v,int f)
{
map[++tt].v=v;
map[tt].f=f;
map[tt].nxt=hd[u];
hd[u]=tt;
} int init(void)
{
n=read(); for (int i=1;i<=n;i++)
{
w[i]=read();
if (w[i]&1)
odd[++odd[0]]=i;
else even[++even[0]]=i;
}
if (odd[0]^even[0]) return 0; for (int i=2;i<P;i++)
{
if (!vp[i]) pri[++pri[0]]=i;
for (int j=1;j<=pri[0];j++)
{
if (i*pri[j]>=P) break;
vp[i*pri[j]]=1;
if (i%pri[j]==0) break;
}
} tt=-1; memset(hd,-1,sizeof hd);
for (int i=1;i<=odd[0];i++)
{
ins(0,odd[i],2),ins(odd[i],0,0);
for (int j=1;j<=even[0];j++)
if (!vp[w[odd[i]]+w[even[j]]])
ins(odd[i],even[j],1),ins(even[j],odd[i],0);
}
for (int i=1;i<=even[0];i++)
ins(even[i],n+1,2),ins(n+1,even[i],0); return 1;
} int BFS(void)
{
memset(level,-1,sizeof level);
h=0,t=1,q[t]=0,level[0]=0; int k;
for (;h^t;)
{
k=q[++h];
for (int r=hd[k];~r;r=map[r].nxt)
if (map[r].f&&level[map[r].v]==-1)
{
level[map[r].v]=level[k]+1;
if (map[r].v==n+1) return 1;
q[++t]=map[r].v;
}
}
return 0;
} inline int min(int i,int j)
{
return i<j?i:j;
} int DFS(int now,int flow)
{
if (now==n+1) return flow;
int sum=0,tmp;
for (int k=hd[now];~k;k=map[k].nxt)
if (map[k].f&&level[now]+1==level[map[k].v])
{
tmp=DFS(map[k].v,min(flow,map[k].f));
if (tmp)
{
map[k].f-=tmp;
map[k^1].f+=tmp;
flow-=tmp;
sum+=tmp;
if (!flow) break;
}
else level[map[k].v]=P;
}
return sum;
} inline void inslist(int now)
{
list[++tl].v=now;
list[tl].nxt=fs[num];
fs[num]=tl;
} void dfs(int now,int fst)
{
cnt[num]++;
used[now]=1;
inslist(now); for (int k=hd[now];k;k=map[k].nxt)
if (!map[k>>1<<1].f&&map[k].v&&map[k].v^n+1)
{
if (fst==map[k].v) return;
if (!used[map[k].v]) dfs(map[k].v,fst);
}
} int work(void)
{
int sum=0;
for (;BFS();) sum+=DFS(0,P);
if (sum^n) return 0; for (int i=1;i<=n;i++)
if (!used[i])
{
num++;
dfs(i,i);
} printf("%d\n",num);
for (int i=1;i<=num;i++)
{
printf("%d ",cnt[i]);
for (int j=fs[i];j;j=list[j].nxt)
printf("%d ",list[j].v);
printf("\n");
} return 1;
} int main(void)
{
int d=init();
if (d) d=work();
if (!d) printf("Impossible\n"); return 0;
}

【Codeforces】512C Fox and Dinner的更多相关文章

  1. 【Codeforces】Round #491 (Div. 2) 总结

    [Codeforces]Round #491 (Div. 2) 总结 这次尴尬了,D题fst,E没有做出来.... 不过还好,rating只掉了30,总体来说比较不稳,下次加油 A:If at fir ...

  2. 【Codeforces】Round #488 (Div. 2) 总结

    [Codeforces]Round #488 (Div. 2) 总结 比较僵硬的一场,还是手速不够,但是作为正式成为竞赛生的第一场比赛还是比较圆满的,起码没有FST,A掉ABCD,总排82,怒涨rat ...

  3. 【CodeForces】601 D. Acyclic Organic Compounds

    [题目]D. Acyclic Organic Compounds [题意]给定一棵带点权树,每个点有一个字符,定义一个结点的字符串数为往下延伸能得到的不重复字符串数,求min(点权+字符串数),n&l ...

  4. 【Codeforces】849D. Rooter's Song

    [算法]模拟 [题意]http://codeforces.com/contest/849/problem/D 给定n个点从x轴或y轴的位置p时间t出发,相遇后按对方路径走,问每个数字撞到墙的位置.(还 ...

  5. 【CodeForces】983 E. NN country 树上倍增+二维数点

    [题目]E. NN country [题意]给定n个点的树和m条链,q次询问一条链(a,b)最少被多少条给定的链覆盖.\(n,m,q \leq 2*10^5\). [算法]树上倍增+二维数点(树状数组 ...

  6. 【CodeForces】925 C.Big Secret 异或

    [题目]C.Big Secret [题意]给定数组b,求重排列b数组使其前缀异或和数组a单调递增.\(n \leq 10^5,1 \leq b_i \leq 2^{60}\). [算法]异或 为了拆位 ...

  7. 【CodeForces】700 D. Huffman Coding on Segment 哈夫曼树+莫队+分块

    [题目]D. Huffman Coding on Segment [题意]给定n个数字,m次询问区间[l,r]的数字的哈夫曼编码总长.1<=n,m,ai<=10^5. [算法]哈夫曼树+莫 ...

  8. 【CodeForces】906 D. Power Tower 扩展欧拉定理

    [题目]D. Power Tower [题意]给定长度为n的正整数序列和模数m,q次询问区间[l,r]累乘幂%m的答案.n,q<=10^5,m,ai<=10^9. [算法]扩展欧拉定理 [ ...

  9. 【CodeForces】741 D. Arpa’s letter-marked tree and Mehrdad’s Dokhtar-kosh paths(dsu on tree)

    [题意]给定n个点的树,每条边有一个小写字母a~v,求每棵子树内的最长回文路径,回文路径定义为路径上所有字母存在一种排列为回文串.n<=5*10^5. [算法]dsu on tree [题解]这 ...

随机推荐

  1. 洛谷 P1313 计算系数 (二项式定理)

    这道题正好复习了二项式定理 所以答案就是a^n * b^m * c(n, k) 然后注意一些细节 我一开始写组合数只写一行的组合数 即c[0] = 1; c[i] = c[i-1] * (n - i ...

  2. 2015 Multi-University Training Contest 8 hdu 5390 tree

    tree Time Limit: 8000ms Memory Limit: 262144KB This problem will be judged on HDU. Original ID: 5390 ...

  3. POJ——T1789 Truck History

    http://poj.org/problem?id=1789 Time Limit: 2000MS   Memory Limit: 65536K Total Submissions: 27597   ...

  4. SharePoint 2010 安装教程

    SharePoint Server 2010作为MOSS 2007的升级版本,自从2009年底发布Beta版本以来就备受关注,网络上已经出现了很多相关的文章,其中也不乏中文的信息. 最近SharePo ...

  5. SQL学习之使用order by 依照指定顺序排序或自己定义顺序排序

    我们通常须要依据客户需求对于查询出来的结果给客户提供自己定义的排序方式,那么我们通常sql须要实现方式都有哪些,參考很多其它资料总结例如以下(不完好的和错误望大家指出): 一.假设我们仅仅是对于在某个 ...

  6. hdu2838Cow Sorting(树状数组+逆序数)

    题目链接:点击打开链接 题意描写叙述:给定一个长度为100000的数组,每一个元素范围在1~100000,且互不同样,交换当中的随意两个数须要花费的代价为两个数之和. 问怎样交换使数组有序.花费的代价 ...

  7. C++线程安全退出

    HANDLE m_EvtThreadExit[MaxVisionNum]; //定义 方法一 ;i<MaxVisionNum;i++) m_EvtThreadExit[index] = Crea ...

  8. oracle中对字符串进行分割,并反回随机段

    --测试数据 select fun_spilt_draw('1,2,3,4,5,6,7') a from dual--待处理数据 strtext--定义一个函数分割 返回字符串中的一个随机段creat ...

  9. VS2012数据绑定控件DataGridView和DataGrid

    在做Windows窗体上ADO.NET数据绑定试验的时候,发现实例中提到的一些控件在vs2012的工具箱中找不到,开始以为是工具箱中的控件太多没看到,结果重新找还是没找到,难道是因为控件升级了?yes ...

  10. HTTP的请求及响应

    前言 本文主要包括以下内容: HTTP是什么? HTTP 请求包括哪些部分? HTTP 响应包括哪些部分? 如何用Chrome开发者工具查看 HTTP 请求及请求的内容? 如何使用 curl 命令? ...