(最大密度子图)

 

poj 3155 (最大密度子图)

分类: 网络流   119人阅读  评论(0)  收藏  举报
ACM 算法 编程 百度 网络流

题意:一个公司有n个人,给出了一些有冲突的人的对数(u,v),公司决定裁人,那么总裁现在要裁掉冲突率最高的那些人(冲突率=在这些人中存在的冲突数/人数)。就是求出一些点,这些点之间的边数/点数最大。最大密度子图。

思路:胡伯涛的论文《最小割模型在信息学竞赛中的应用》介绍了两种方法:

第一种:转换为最大权闭合图的模型来求解:

设max g = f(x)= |E‘|/|V’| ,找一个子图的边数与点数的比值达到其中的最大,我们通常都是构造一个函数max h(g)= |E'|-g*|V'|,当h(g)为0的时候,g的值即为最优,h(g)>0 时 g<最优值, h(g)<0时,g>最优值;因为如果最大值大于0那么我们就可以继续增加g的值来减小h(g),若最大值都小于0了,那么g不可能增加只可能减少!

观察h(g),边和点有依赖关系,就边依赖点,边存在的必要条件是点的存在,那么这样以后,如果我们将边看成点,那么这不就符合最大权闭合子图了。现在h(g)的求法就可以通过求新图的最大权闭合子图的值来求解,但是这里有个问题,建图之后你可以发现当求出来的值和h(g)原本应该为值不对应(具体为什么不怎么理解),可以这样理解,当最小的一个g使得h(g)为0的时候该解即为最优解,因为h(g)是以个单调递减函数,就该函数来看只可能存在一个g使得h(g)=0;然而通过求最大权闭合子图是子图权值和为0的有很多中g,当最小的一个g使得h(g)为0之后,如果g继续增大那么虽然通过最大权闭合子图的值求出来依旧为0,但是真正的h(g)< 0 了,所以要使得最优的一个解就是使得最大权闭合子图的权值和为0的最小的一个g值!这样求解之后从源点流到汇点为满流的边即为最大密度子图中的点。

第二种:

源点到各个点连接一条有向边权值为U,各个点到汇点连接一条边权值为U+2*g-d,原来有关系的点连接两条有向边(u,v),(v,u)权值为1(U可以取m,U的目的是用来使得2*g-d的值始终为正),这样以后求最小割,那么h(g)= (U*n-mincut)/2;二分找到最优值即为mid ,但是如果要求图中的点则需要用left来从新图求最大流之后然后从源点开始dfs遍历,最后得出结果。


第一种代码:


[cpp]  view plain copy
  1. #include<stdio.h>  
  2. #include<string.h>  
  3. const int N=1500;  
  4. const double inf=0x3fffffff;  
  5. const double eps=1e-8;  
  6. int gap[N],dis[N],start,end,ans,sum,head[N],num,dep[N],n,m;  
  7. bool vis[N];  
  8. struct edge  
  9. {  
  10.     int st,ed,next;  
  11.     double flow;  
  12. }e[80*N];  
  13. struct node  
  14. {  
  15.     int x,y;  
  16. }P[1100];  
  17. void addedge(int x,int y,double w)  
  18. {  
  19.     e[num].st=x;e[num].ed=y;e[num].flow=w;e[num].next=head[x];head[x]=num++;  
  20.     e[num].st=y;e[num].ed=x;e[num].flow=0;e[num].next=head[y];head[y]=num++;  
  21. }  
  22. void makemap(double g)  
  23. {  
  24.     int i;  
  25.     memset(head,-1,sizeof(head));  
  26.     num=0;  
  27.     for(i=1;i<=n;i++)  
  28.         addedge(i,end,g);  
  29.     for(i=0;i<m;i++)  
  30.     {  
  31.         addedge(n+i+1,P[i].y,inf);  
  32.         addedge(n+i+1,P[i].x,inf);  
  33.         addedge(start,n+i+1,1.0);  
  34.     }  
  35. }  
  36. double dfs(int u,double minflow)    
  37. {    
  38.     if(u==end)return minflow;    
  39.     int i,v;    
  40.     double f,flow=0.0;    
  41.     for(i=head[u];i!=-1;i=e[i].next)    
  42.     {    
  43.         v=e[i].ed;    
  44.         if(e[i].flow>0)    
  45.         {    
  46.             if(dis[v]+1==dis[u])    
  47.             {    
  48.                 f=dfs(v,e[i].flow>minflow-flow?minflow-flow:e[i].flow);    
  49.                 flow+=f;    
  50.                 e[i].flow-=f;    
  51.                 e[i^1].flow+=f;    
  52.                 if(minflow-flow<=1e-8)return flow;      
  53.                 if(dis[start]>=ans)return flow;      
  54.             }    
  55.         }    
  56.     }    
  57.     if(--gap[dis[u]]==0)    
  58.         dis[start]=ans;    
  59.     dis[u]++;    
  60.     gap[dis[u]]++;    
  61.     return flow;    
  62. }    
  63. double isap()    
  64. {    
  65.     double maxflow=0.0;    
  66.     memset(gap,0,sizeof(gap));    
  67.     memset(dis,0,sizeof(dis));    
  68.     gap[0]=ans;    
  69.     while(dis[start]<ans)    
  70.         maxflow+=dfs(start,inf);    
  71.     return 1.0*m-maxflow;     
  72. }  
  73. void dfs1(int u)  
  74. {  
  75.     vis[u]=true;  
  76.     if(u>=1&&u<=n)  
  77.     sum++;  
  78.     for(int i=head[u];i!=-1;i=e[i].next)  
  79.     {  
  80.         int v=e[i].ed;  
  81.         if(vis[v]==false&&e[i].flow>0)  
  82.           dfs1(v);  
  83.     }  
  84. }  
  85. int main()  
  86. {  
  87.     int i;  
  88.     double Left,Right,mid,flow;  
  89.     while(scanf("%d%d",&n,&m)!=-1)  
  90.     {  
  91.         if(m==0){printf("1\n1\n");continue;}  
  92.         start=0,end=n+m+1,ans=end+1;  
  93.         for(i=0;i<m;i++)  
  94.         {  
  95.             scanf("%d%d",&P[i].x,&P[i].y);  
  96.         }  
  97.         Left=0;Right=m;  
  98.         while(Right-Left>=1.0/n/n)//胡伯涛的论文给出了证明,不同解之间误差的精度不超过1/(n*n)    
  99.         {  
  100.             mid=(Left+Right)/2;  
  101.             makemap(mid);  
  102.             flow=isap();//求出最大权值闭合图  
  103.             if(flow<eps)//如果小于0,g值太大  
  104.                 Right=mid;  
  105.             else Left=mid;  
  106.         }  
  107.         makemap(Left);//最大密度建图  
  108.         isap();  
  109.         memset(vis,false,sizeof(vis));  
  110.         sum=0;  
  111.         dfs1(start);  
  112.         printf("%d\n",sum);  
  113.         for(i=1;i<=n;i++)  
  114.           if(vis[i]==true)//残留网络中源点能到达的点  
  115.               printf("%d\n",i);  
  116.     }  
  117.     return 0;  
  118. }  

第二种代码:


[cpp]  view plain copy
  1. #include<stdio.h>  
  2. #include<string.h>  
  3. const int N=110;  
  4. const double inf=0x3fffffff;  
  5. const double eps=1e-8;  
  6. int gap[N],dis[N],start,end,ans,sum,head[N],num,dep[N],n,m;  
  7. bool vis[N];  
  8. struct edge  
  9. {  
  10.     int st,ed,next;  
  11.     double flow;  
  12. }e[80*N];  
  13. struct node  
  14. {  
  15.     int x,y;  
  16. }P[1100];  
  17. void addedge(int x,int y,double w)  
  18. {  
  19.     e[num].st=x;e[num].ed=y;e[num].flow=w;e[num].next=head[x];head[x]=num++;  
  20.     e[num].st=y;e[num].ed=x;e[num].flow=0;e[num].next=head[y];head[y]=num++;  
  21. }  
  22. void makemap(double g)  
  23. {  
  24.     int i;  
  25.     memset(head,-1,sizeof(head));  
  26.     num=0;  
  27.     for(i=1;i<=n;i++)  
  28.     {  
  29.         addedge(start,i,m*1.0);  
  30.         addedge(i,end,m+2*g-dep[i]);  
  31.     }  
  32.     for(i=0;i<m;i++)  
  33.     {  
  34.         addedge(P[i].x,P[i].y,1.0);  
  35.         addedge(P[i].y,P[i].x,1.0);  
  36.     }  
  37. }  
  38. double dfs(int u,double minflow)    
  39. {    
  40.     if(u==end)return minflow;    
  41.     int i,v;    
  42.     double f,flow=0.0;    
  43.     for(i=head[u];i!=-1;i=e[i].next)    
  44.     {    
  45.         v=e[i].ed;    
  46.         if(e[i].flow>0)    
  47.         {    
  48.             if(dis[v]+1==dis[u])    
  49.             {    
  50.                 f=dfs(v,e[i].flow>minflow-flow?minflow-flow:e[i].flow);    
  51.                 flow+=f;    
  52.                 e[i].flow-=f;    
  53.                 e[i^1].flow+=f;    
  54.                 if(minflow-flow<=1e-8)return flow;      
  55.                 if(dis[start]>=ans)return flow;      
  56.             }    
  57.         }    
  58.     }    
  59.     if(--gap[dis[u]]==0)    
  60.         dis[start]=ans;    
  61.     dis[u]++;    
  62.     gap[dis[u]]++;    
  63.     return flow;    
  64. }    
  65. double isap()    
  66. {    
  67.     double maxflow=0.0;    
  68.     memset(gap,0,sizeof(gap));    
  69.     memset(dis,0,sizeof(dis));    
  70.     gap[0]=ans;    
  71.     while(dis[start]<ans)    
  72.         maxflow+=dfs(start,inf);    
  73.     return maxflow;     
  74. }  
  75. void dfs1(int u)//遍历要选的点  
  76. {  
  77.     vis[u]=true;  
  78.     sum++;  
  79.     for(int i=head[u];i!=-1;i=e[i].next)  
  80.     {  
  81.         int v=e[i].ed;  
  82.         if(vis[v]==false&&e[i].flow>0)  
  83.           dfs1(v);  
  84.     }  
  85. }  
  86. int main()  
  87. {  
  88.     int i;  
  89.     double Left,Right,mid,hg;  
  90.     while(scanf("%d%d",&n,&m)!=-1)  
  91.     {  
  92.         if(m==0){printf("1\n1\n");continue;}  
  93.         start=0,end=n+1,ans=end+1;  
  94.         memset(dep,0,sizeof(dep));  
  95.         for(i=0;i<m;i++)  
  96.         {  
  97.             scanf("%d%d",&P[i].x,&P[i].y);  
  98.             dep[P[i].x]++;dep[P[i].y]++;  
  99.         }  
  100.         Left=0;Right=m;  
  101.         while(Right-Left>=1.0/n/n)//胡伯涛的论文给出了证明,不同解之间误差的精度不超过1/(n*n)  
  102.         {  
  103.             mid=(Left+Right)/2;  
  104.             makemap(mid);  
  105.             hg=isap();  
  106.             hg=(1.0*n*m-hg)/2;  
  107.             if(hg>eps)  
  108.                 Left=mid;  
  109.             else Right=mid;  
  110.         }  
  111.         makemap(Left);//用mid值建图容易wa,因为你此时的mid不一定满足h(mid)>eps,但是Left一定是满足的  
  112.         isap();  
  113.         memset(vis,false,sizeof(vis));  
  114.         sum=0;  
  115.         dfs1(0);  
  116.         printf("%d\n",sum-1);  
  117.         for(i=1;i<=n;i++)  
  118.           if(vis[i]==true)  
  119.               printf("%d\n",i);  
  120.     }  
  121.     return 0;  
  122. }  

你可能感兴趣的:(网络流)