POJ 2989

题意:给定一个无向图(节点数小于128)求极大团(不包含在更大的团中)的总数。

对最大团,极大团不熟悉的,建议先阅读最大团搜索问题 ZOJ 1492 再来看本题。

本题数据有限,可以使用dfs解决。类似于搜索最大团的加强版。有“两个”剪枝需要注意。

在dfs中需要维护两个集合即 Not(已经尝试过搜索极大团的节点)和Candidate(未曾尝试过的节点)由于极大团带有集合的性质,故某个节点在dfs序列中出现的位次并不重要。

当Not和Can集合同时为空时结束dfs。

在代码中,Not节点集合用ne数组标记,Can节点集合用ce数组标记

剪枝1:若当前Not集合中存在一个点,与Can中所有点都相连,则在未来的搜索中它永远不会离开Not集合,故剪枝。

剪枝2:这个剪枝为了优化剪枝1的效果,并不是一个新的剪枝。

    原本的搜索我们总是从Can集合中随意选择一个继续dfs 但是我们可以挑选一个特殊的节点,来增强剪枝1的效果。

设每个在Not集合中的节点有一个cnt值,为Can集合中与它不相连的节点个数。

我们于是选择Can集合中与cnt最小的节点不相连的节点进行下一步dfs。

代码如下

#include<iostream>
#include<cstdio>
#include<cstdlib>
#include<cstring>
#include<cmath>
#include<vector>
#include<algorithm>
#include<queue>
using namespace std; const int maxn=130;
int n,m,ans,ne[maxn],ce[maxn],list[maxn][maxn];
bool g[maxn][maxn];
void dfs(int size)
{
if(ans>1000)return ;
int i,j,k,t,cnt,best=0;
if(ne[size]==ce[size])
{
if(ce[size]==0)++ans;
return ;
}
for(t=0,i=1;i<=ne[size];++i)
{
for(cnt=0,j=ne[size]+1;j<=ce[size];++j)
if(!g[list[size][i]][list[size][j]])++cnt;
if(t==0||cnt<best)t=i,best=cnt;
}
if(t&&best<=0)return ;//剪枝1
for(k=ne[size]+1;k<=ce[size];++k)
{
if(t>0)
{
for(i=k;i<=ce[size];++i)
if(!g[list[size][t]][list[size][i]])break;
swap(list[size][k],list[size][i]);//最大化剪枝1
}
i=list[size][k];
ne[size+1]=ce[size+1]=0;
for(j=1;j<k;++j)
if(g[i][list[size][j]])
list[size+1][++ne[size+1]]=list[size][j];
for(ce[size+1]=ne[size+1],j=k+1;j<=ce[size];++j)
if(g[i][list[size][j]])
list[size+1][++ce[size+1]]=list[size][j];
dfs(size+1);
if(ans>1000)return ;
++ne[size];
--best;
for(j=k+1,cnt=0;j<=ce[size];++j)
if(!g[i][list[size][j]])
++cnt;
if(t==0||cnt<best)t=k,best=cnt;
if(t&&best<=0)break;
}
} void cluster_count()
{
int i;
ne[0]=0;ce[0]=0;
for(i=1;i<=n;++i)
list[0][++ce[0]]=i;
ans=0;
dfs(0);
} int main()
{
while(scanf("%d%d",&n,&m)!=EOF)
{
memset(g,0,sizeof(g));
for(int i=0;i<m;i++)
{
int a,b;
scanf("%d%d",&a,&b);
g[a][b]=g[b][a]=true;
}
cluster_count();
if(ans>1000)printf("Too many maximal sets of friends.\n");
else printf("%d\n",ans);
}
return 0;
}

  

POJ 2989 All Friends 极大团计数的更多相关文章

  1. All Friends 极大团

    搞懂了什么是团  什么是极大团  什么是最大团 极大团就是  不是任何团的真子集  最大团就是点数最多的极大团 这题就是求极大团的个数 用bk算法 #include <iostream> ...

  2. hdoj1373 Channel Allocation(极大团)

    题意是有若干个接收器,给出每个接收器的相邻接收器.相邻的接收器不能使用同一信号频道.问所需要的信号频道数. 求该无向图的极大团. #include<iostream> #include&l ...

  3. Codeforces 839E Mother of Dragons(极大团)

    [题目链接] http://codeforces.com/contest/839/problem/E [题目大意] 现在有一些点,现在你有k的液体,随意分配给这些点, 当两个点有边相连的时候,他们能产 ...

  4. POJ 3692 Kindergarten (二分图 最大团)

    Kindergarten Time Limit: 2000MS   Memory Limit: 65536K Total Submissions: 5660   Accepted: 2756 Desc ...

  5. POJ 3692 Kindergarten(最大团问题)

    题目链接:http://poj.org/problem?id=3692 Description In a kindergarten, there are a lot of kids. All girl ...

  6. POJ 2409 Let it Bead(polay计数)

    题目链接:http://poj.org/problem?id=2409 题意:给出一个长度为m的项链,每个珠子可以用n种颜色涂色.翻转和旋转后相同的算作一种.有多少种不同的项链? 思路: (1) 对于 ...

  7. poj 1952 最长公共子序列计数

    看代码就懂了  不解释  3 1 1 1 1 2 2 2 1 1 1 3  第一个3 和最后一个 3 只需要一个就够了,,, #include<iostream> #include< ...

  8. [ACM] POJ 2409 Let it Bead (Polya计数)

    参考:https://blog.csdn.net/sr_19930829/article/details/38108871 #include <iostream> #include < ...

  9. poj 2409 Let it Bead Polya计数

    旋转能够分为n种置换,相应的循环个数各自是gcd(n,i),个i=0时不动,有n个 翻转分为奇偶讨论,奇数时有n种置换,每种有n/2+1个 偶数时有n种置换,一半是n/2+1个,一半是n/2个 啃论文 ...

随机推荐

  1. AJAX异步加载

    AJAX含义: 即"Asynchronous Javascript And XML"(异步JavaScript和XML),是指一种创建交互式网页应用的网页开发技术. AJAX可以跨 ...

  2. OpenSceneGraph几个重要功能节点练习

    OpenSceneGraph几个重要功能节点练习 一. 空间变换节点 空间变换中最重要的是坐标系和矩阵运算了.OSG坐标系中使用右手系,Z轴垂直向上,X轴水平向右,Y轴垂直屏幕向里,与OpenGL和D ...

  3. ReactiveCocoa 设置绑定注意事项

    要在ViewLoad里面进行绑定,因为在init中还没有初始化界面,所有绑定会无效

  4. PHP新手之学习数组声明

    数组是在程序设计中,为了处理方便, 把具有相同类型的若干变量按有序的形式组织起来的一种形式.这些按序排列的同类数据元素的集合称为数组.下面介绍PHP中的数组声明. 一.数组的概述 1.数组的本质:管理 ...

  5. WEB 开发者应该具备的 6 大技能?

    1. 界面和用户体验 注意,浏览器的实现标准是不一致的,请确保你的网站能够兼容所有主流的浏览器.最少需要测试的有 Gecko 引擎 (Firefox),WebKit引擎(Safari以及一些手机浏览器 ...

  6. apache的工作模式 和 最大连接数设置

    经过测试 效果明显 (1)首选查看apache的工作模式 windows下的查看apache的工作模式命令:httpd -l 如果列出mod_win32.c,则表示是 win32.c 工作方式. 列出 ...

  7. bmp图片显示

    文件IO项目: 在开发板屏幕上循环显示目录里的图片 a.按照一定的间隔循环显示目录里的bmp图片 b.实现手指滑动来显示目录里的图片(bmp,jpg)上一张,下一张 d1: 1.能操控屏幕(查询开发板 ...

  8. MariaDB与MySQL在一台服务器同时运行

    [root@HE3 ~]#groupadd mariadb -g 513 [root@HE3 ~]#useradd-u 513 -gmariadb -s /sbin/nologin -d /home/ ...

  9. 清除delphi 控件DBgrid 的记录

    http://blog.csdn.net/windhaunting/article/details/4751560 1.TTable(DBGrid1.DataSource.DataSet).Empty ...

  10. 如何使用Ninja快速编译LLVM和Clang

    在使用Make工具编译LLVM是非常耗时的.往往需要三四个小时.但是使用goolge开源的ninja编译LLVM只需要10到20分钟. 本文以llvm3.3为例,演示在linux上编译和安装过程. 第 ...