POJ 2531 Network Saboteur (枚举+剪枝)
题意:给你一个图,图中点之间会有边权,现在问题是把图分成两部分,使得两部分之间边权之和最大。
目前我所知道的有四种做法:
方法一:状态压缩
#include <iostream>
#include <stdio.h>
#include <string.h>
#include <algorithm>
/*
用状态压缩枚举,297ms。 题意:给你一个图,图中点之间会有边权,现在问题是把图分成两部分,使得两部分之间边权之和最大。 情况是对称的,也就是说枚举所有的分类情况中,有一半是冗余的。
举个例子来说,有5个节点,1、2、3分配为A组,4、5分配给B组,
这个分类的策略所计算的结果和1、2、3分配为B组,4、5分配给A组是一样的。
因此对称的就不需要再考虑了,如状态压缩时101和010,只要考虑其中一个就可以了。
至于当枚举到一个状态时,怎么判断它的对称状态已经枚举过了,采用异或就可以了。 */
using namespace std;
const int maxn=;
int n;
int w[maxn][maxn];
//int vis[maxn];
int visit[*];
int ans=;
int set1[maxn];
int set2[maxn];
int idx1,idx2;
int main()
{
int l;
//cout<<(1<<20)<<endl;
cin>>n; memset(w,,sizeof(w));
for(int i=;i<=n;i++){
for(int j=;j<=i;j++)
scanf("%d",&l);
for(int j=i+;j<=n;j++){
scanf("%d",&l);
w[i][j]=w[j][i]=l;
}
}
memset(vis,,sizeof(vis));
memset(visit,,sizeof(visit));
int all=<<n-;
ans=;
//用状态压缩来枚举两组的情况
for(int i=;i<(<<n)-;i++){
//对称的就不需要再考虑了,如101和010,只要考虑其中一个就可以了,用异或即可求出对称的状态。
if(!visit[all^i]){
visit[i]=;
idx1=idx2=;
//将编号存入两个数组中去,而不是用vis标记,从原本的1000+ms减少到297ms
for(int j=;j<n;j++){
if(i&(<<j))
set1[idx1++]=j+;
else
set2[idx2++]=j+;
}
int tot=;
for(int k=;k<idx1;k++){
for(int t=;t<idx2;t++){
tot+=w[set1[k]][set2[t]];
}
}
if(tot>ans)
ans=tot;
}
}
printf("%d\n",ans);
return ;
}
方法二:dfs枚举
#include <iostream>
#include <stdio.h>
#include <string.h>
#include <algorithm>
/*
AC
dfs暴力枚举 547ms
*/
using namespace std;
const int maxn=;
int n;
int w[maxn][maxn];
int vis[maxn];
int set1[maxn];
int set2[maxn];
int idx1,idx2;
int ans=;
void dfs(int u){
if(u==n+){
int tot=;
idx1=idx2=;
for(int i=;i<=n;i++){
if(vis[i])
set1[idx1++]=i;
else
set2[idx2++]=i;
}
for(int k=;k<idx1;k++){
for(int t=;t<idx2;t++){
tot+=w[set1[k]][set2[t]];
}
}
if(tot>ans)
ans=tot;
return ;
}
vis[u]=;
/*
之前还用for循环,额,脑残了。。。
for(int i=u+1;i<=n+1;i++){
dfs(i);
}
*/
dfs(u+);
vis[u]=;
dfs(u+); }
int main()
{
int l;
cin>>n;
//cout<<(1<<19)<<endl;
memset(w,,sizeof(w));
for(int i=;i<=n;i++){
for(int j=;j<=i;j++)
scanf("%d",&l);
for(int j=i+;j<=n;j++){
scanf("%d",&l);
w[i][j]=w[j][i]=l;
}
}
memset(vis,,sizeof(vis));
dfs();
printf("%d\n",ans);
return ;
}
方法三:大牛的解法
#include <iostream>
#include <stdio.h>
#include <string.h>
#include <algorithm>
/*
看了discuss里大牛的做法,然后自己打了一遍,16ms
以下来自discuss里的: 1.等价剪枝:根据对称性剪枝
举个例子来说,有5个节点,1、2、3分配为A组,4、5分配给B组,
这个分类的策略所计算的结果和1、2、3分配为B组,4、5分配给A组是一样的。
那么怎么去除这一半的冗余呢,其实很简单,我们强制第1个节点分配给A组,
那么后面不管怎么分,全体情况都不会有重复的了。 2.换个角度,更高效搜索算法 :如本题逆向思维求最小内部代价
本题要求两个集合之间的边权和最大。反过来,相当于两个集合内部点之间的边权和最小,
令该和为ans。那么我们dfs的目的就是使得ans的值最小。 3.无法达最优:提前剪枝,也就是如果当前的Sum>=ans,就不必继续下去了
*/
using namespace std;
const int maxn=;
int n;
int w[maxn][maxn];
int set1[maxn];
int set2[maxn];
int idx1,idx2;
int tot1,tot2,tot;
int ans; void dfs(int Sum,int idx1,int idx2,int k){
if(Sum>=ans)
return;
else if(k==n+){
ans=Sum;
}
else{
int sum=;
for(int i=;i<idx1;i++){
sum+=w[set1[i]][k];
}
set1[idx1]=k; //若k属于第一个集合
dfs(Sum+sum,idx1+,idx2,k+); sum=;
for(int i=;i<idx2;i++){
sum+=w[set2[i]][k];
}
set2[idx2]=k; //若k属于第二个集合
dfs(Sum+sum,idx1,idx2+,k+);
}
}
int main()
{
int l;
cin>>n;
memset(w,,sizeof(w));
tot=;
for(int i=;i<=n;i++){
for(int j=;j<=i;j++)
scanf("%d",&l);
for(int j=i+;j<=n;j++){
scanf("%d",&l);
w[i][j]=w[j][i]=l;
tot+=l;
}
}
tot1=;
for(int i=;i<n/;i++){
for(int j=i+;j<n/;j++)
tot1+=w[i][j];
}
tot2=;
for(int i=n/;i<=n;i++){
for(int j=i+;j<=n;j++)
tot2+=w[i][j];
}
ans=tot1+tot2;
idx1=idx2=;
set1[]=; //先把1固定为第一个集合
dfs(,,,);
printf("%d\n",tot-ans);
return ;
}
方法四:随机算法
http://www.cnblogs.com/chujian123/p/3533156.html
POJ 2531 Network Saboteur (枚举+剪枝)的更多相关文章
- POJ 2531 Network Saboteur 位运算子集枚举
题目: http://poj.org/problem?id=2531 这个题虽然是个最大割问题,但是分到dfs里了,因为节点数较少.. 我试着位运算枚举了一下,开始超时了,剪了下枝,1079MS过了. ...
- POJ 2531 Network Saboteur
http://poj.org/problem?id=2531 题意 :有N台电脑,每两台电脑之间都有个通信量C[i][j]; 问如何将其分成两个子网,能使得子网之间的通信量最大. 也就是说将所有节点分 ...
- poj 2531 Network Saboteur 解题报告
题目链接:http://poj.org/problem?id=2531 题目意思:将 n 个点分成两个部分A和B(也就是两个子集啦), 使得子集和最大(一定很难理解吧,呵呵).举个例子吧,对于样例,最 ...
- poj 2531 Network Saboteur( dfs )
题目:http://poj.org/problem?id=2531 题意:一个矩阵,分成两个集合,求最大的 阻碍量 改的 一位大神的代码,比较简洁 #include<stdio.h> #i ...
- PKU 2531 Network Saboteur(dfs+剪枝||随机化算法)
题目大意:原题链接 给定n个节点,任意两个节点之间有权值,把这n个节点分成A,B两个集合,使得A集合中的每一节点与B集合中的每一节点两两结合(即有|A|*|B|种结合方式)权值之和最大. 标记:A集合 ...
- poj 2531 Network Saboteur(经典dfs)
题目大意:有n个点,把这些点分别放到两个集合里,在两个集合的每个点之间都会有权值,求可能形成的最大权值. 思路:1.把这两个集合标记为0和1,先默认所有点都在集合0里. 2 ...
- poj 2531 搜索剪枝
Network Saboteur Time Limit: 2000 MS Memory Limit: 65536 KB 64-bit integer IO format: %I64d , %I64u ...
- Network Saboteur 分类: 搜索 POJ 2015-08-09 19:48 7人阅读 评论(0) 收藏
Network Saboteur Time Limit: 2000MS Memory Limit: 65536K Total Submissions: 10147 Accepted: 4849 Des ...
- poj2531 Network Saboteur
Network Saboteur Time Limit: 2000MS Memory Limit: 65536K Total Submissions: 11122 Accepted: 5372 ...
随机推荐
- 动态加载故事storyboard
第一种方法: ViewControllerB *vcB = [self.storyboard instantiateViewControllerWithIdentifier:@"Name o ...
- SQL基础知识----数据类型
VARCHAR(VERiable CHARacter):可变动字符.用于保存以文本格式处处的信息,最大可以储存255个字符.一般使用为VAECHAR(10) --表示最多可以存储10个字符 INT ...
- 深度理解依赖注入(Dependence Injection)
前面的话:提到依赖注入,大家都会想到老马那篇经典的文章.其实,本文就是相当于对那篇文章的解读.所以,如果您对原文已经有了非常深刻的理解,完全不需要再看此文:但是,如果您和笔者一样,以前曾经看过,似乎看 ...
- HTML5的placeholder属性如何实现换行
在HTML5中,placeholder是一个非常有用的属性,当控件中无内容时可以代替UI控件的提示功能,而不需要写额外的代码.但如果有一个textarea控件,我们需要多行的文本提示信息时,使用”\n ...
- ToolBar存档
上图是将本阶段要完成的结果画面做了标示,结合下面的描述希望大家能明白. colorPrimaryDark(状态栏底色):在风格 (styles) 或是主题 (themes) 里进行设定. App ba ...
- unity工具IGamesTools之批量生成帧动画
unity工具IGamesTools批量生成帧动画,可批量的将指定文件夹下的帧动画图片自动生成对应的资源文件(Animation,AnimationController,Prefabs) unity工 ...
- PHP URL 重定向 的三种方法(转载)
为了方便查询,转载一篇. 1.使用header()函数 PHP的HTTP相关函数种提供了一个 header()函数,首先要清楚,header()函数必须放在php程序的开头部分,而且之前不能有另 ...
- java笔试题(3)
short a = 1; a = a + 1; 有错吗? short a = 1; a += 1; 有错吗? 对于short a = 1; a = a + 1;由于a + 1 运算时会自动提升表达式的 ...
- Mysql高级之存储过程
参考地址1:http://www.2cto.com/database/201411/350819.html 参考地址2:http://www.jb51.net/article/39471.htm my ...
- 网页数据采集 - 系列之Flash数据采集
经常看到一些朋友在讨论如何采集flash中的数据,讨论来讨论区,结论就是:flash不能采集,其实也不总是这样.本篇就跟大家分享如何采集flash中的数据. 在开始之前,先说明一下:一般来说flash ...