题意:给你一个图,图中点之间会有边权,现在问题是把图分成两部分,使得两部分之间边权之和最大。

目前我所知道的有四种做法:

方法一:状态压缩

#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 (枚举+剪枝)的更多相关文章

  1. POJ 2531 Network Saboteur 位运算子集枚举

    题目: http://poj.org/problem?id=2531 这个题虽然是个最大割问题,但是分到dfs里了,因为节点数较少.. 我试着位运算枚举了一下,开始超时了,剪了下枝,1079MS过了. ...

  2. POJ 2531 Network Saboteur

    http://poj.org/problem?id=2531 题意 :有N台电脑,每两台电脑之间都有个通信量C[i][j]; 问如何将其分成两个子网,能使得子网之间的通信量最大. 也就是说将所有节点分 ...

  3. poj 2531 Network Saboteur 解题报告

    题目链接:http://poj.org/problem?id=2531 题目意思:将 n 个点分成两个部分A和B(也就是两个子集啦), 使得子集和最大(一定很难理解吧,呵呵).举个例子吧,对于样例,最 ...

  4. poj 2531 Network Saboteur( dfs )

    题目:http://poj.org/problem?id=2531 题意:一个矩阵,分成两个集合,求最大的 阻碍量 改的 一位大神的代码,比较简洁 #include<stdio.h> #i ...

  5. PKU 2531 Network Saboteur(dfs+剪枝||随机化算法)

    题目大意:原题链接 给定n个节点,任意两个节点之间有权值,把这n个节点分成A,B两个集合,使得A集合中的每一节点与B集合中的每一节点两两结合(即有|A|*|B|种结合方式)权值之和最大. 标记:A集合 ...

  6. poj 2531 Network Saboteur(经典dfs)

    题目大意:有n个点,把这些点分别放到两个集合里,在两个集合的每个点之间都会有权值,求可能形成的最大权值.   思路:1.把这两个集合标记为0和1,先默认所有点都在集合0里.             2 ...

  7. poj 2531 搜索剪枝

    Network Saboteur Time Limit: 2000 MS Memory Limit: 65536 KB 64-bit integer IO format: %I64d , %I64u ...

  8. Network Saboteur 分类: 搜索 POJ 2015-08-09 19:48 7人阅读 评论(0) 收藏

    Network Saboteur Time Limit: 2000MS Memory Limit: 65536K Total Submissions: 10147 Accepted: 4849 Des ...

  9. poj2531 Network Saboteur

    Network Saboteur Time Limit: 2000MS   Memory Limit: 65536K Total Submissions: 11122   Accepted: 5372 ...

随机推荐

  1. oracle 11g 表空间使用率

    Oracle数据库表空间使用量查询: select b.file_name 物理文件名,b.tablespace_name 表空间,b.bytes/1024/1024 大小M,(b.bytes-sum ...

  2. Apache+tomcat集群配置

    一.软件准备 Apache 2.2 : http://httpd.apache.org/download.cgi,下载msi安装程序,选择no ssl版本 Tomcat 6.0 : http://to ...

  3. WebClient和HttpReuqest两种网络请求的方式

    相对来说webClient请求的方式比较简单,可以直接通过new的方式创建一个实例,然后调用OpenReadAsync方法传进一个url,最后通过回调函数OpenReadCompleted就可以获取网 ...

  4. IOS中 如何去除Tabview里面cell之间的下划线

    可以利用Tabview的separatorStyle属性来设置,选择其中的UITableViewCellSeparatorStyleNone 即可去除cell之间的下划线 self.tableView ...

  5. C# Hadoop学习笔记

    记录一下学习地址 http://www.360doc.com/content/14/0607/22/3218170_384675141.shtml

  6. 【EF Code First】 一对多、多对多的多重关系配置

    这里使用用户表(User)和项目(Project)表做示例 有这样一个需求: 用户与项目的关系是:一个用户可以发多个项目,可以参加多个项目,而项目可以有多个参与成员和一个发布者 [其中含1-n和n-n ...

  7. html 元素 变小手

    要设置鼠标指针放在一个元素边界范围内时所用的光标形状,需要对元素的css属性cursor进行设置.cursor属性可能的值1.default 默认光标(通常是一个箭头)2.auto 默认.浏览器设置的 ...

  8. UIProgressView swift

    // // ViewController.swift // UILabelTest // // Created by mac on 15/6/23. // Copyright (c) 2015年 fa ...

  9. 20145120 《Java程序设计》第8周学习总结

    20145120 <Java程序设计>第8周学习总结 教材学习内容总结 NIO使用频道(channel)来衔接数据节点 read()将ReadableByteChannel中的数据读至By ...

  10. Tomcat配置HTTPS方式生成安全证书

    在Tomcat 6中配置SSL双向认证是相当容易的,本文将介绍如何使用JDK的keytool来为Tomcat配置双向SSL认证.并实现批量生成证书 系统需求:JDK 5.0Tomcat 6.0.16启 ...