Codeforces 566E - Restoring Map(bitset 优化构造)
本来说好的不做,结果今早又忍不住开了道题/qiao
我们称度为 \(1\) 的点为叶节点,度大于 \(1\) 的点为非叶节点。
首先考虑如何求出叶节点及其连边情况,这里不妨假设叶节点个数 \(\ge 3\),对于 \(\le 2\) 的情况特判掉,具体如何特判见下文。可以发现,对于两个非叶节点 \(x,y\),如果它们之间存在边相连,那么就一定存在两个点,到它们之间距离 \(\le 2\) 的点的集合恰好是 \(\{x,y\}\),具体构造就是取一个与 \(x\) 相连且 \(\ne y\) 的点,再取一个与 \(y\) 相连且 \(\ne x\) 的点。而如果存在两个点,满足它们交集恰好是 \(\{x,y\}\),那 \(x,y\) 之间必然存在边相连,否则如果 \(x,y\) 距离为 \(2\),那么这两点的交集中如果包含 \(x,y\),那么必然包含 \(x,y\) 的公共邻居,也就是 \(x,y\) 中间夹着的那个点,如果 \(x,y\) 距离为 \(3\),那么道理也是类似的,如果这两点的交集中如果包含 \(x,y\),那么必然包含 \(x,y\) 的之间的两个点,如果 \(x,y\) 距离 \(\ge 4\) 那么不可能存在两个点,到它们距离 \(\le 2\) 的点集中都包含 \(x,y\)。这个可以使用 bitset
及其自带 _Find_first
和 _Find_next
函数解决。
这样我们可以知道非叶节点之间的连边情况,接下来考虑如何将叶子挂上去。首先思考如何判断一个点是否对应一个叶节点的集合,显然对于一个非叶节点 \(x\) 及一个与其相连的叶节点 \(y\),必然有 \(y\) 对应的集合完全包含在 \(x\) 对应的集合中,这样我们遍历到一个点时,我们检验是否存在另一个集合完全包含在该集合中,如果有则跳过这个集合。这样可以保证我们遍历到的所有集合要么属于一个叶节点,要么属于一个不与任何点相连的非叶节点,对于后者而言显然不会对答案产生任何影响,因此我们不跳过它也罢。
接下来考虑如何判定一个叶节点的集合连向的是哪个非叶节点,我们去掉该集合中所有叶子节点(显然在第一步中我们已经知道哪些是叶子节点,而哪些不是),那么可以发现,去掉叶子节点后的集合,等于所有和与其相连的非叶节点的非叶节点组成的集合,而对于每个非叶节点,我们是知道哪些非叶节点与其相连的,这个同样可以 bitset
优化。
接下来找出这个非叶节点之后我们就可以知道所有与这个非叶节点相邻的叶节点了。具体方法就是找到这个集合中所有叶节点,显然这些叶节点都与这个非叶节点相邻。
这样我们就完美地处理了非叶节点个数 \(\ge 3\) 的情况,那如果非叶节点个数 \(\le 3\) 怎么办呢?
显然如果非叶节点个数 \(=1\),那么答案就是个菊花图,随便构造一个菊花图就行了。
如果非叶节点个数 \(=2\),那么所有叶子节点对应的集合大小都 \(<n\),而这种情况连向什么非叶节点是不重要的,因此我们仿照上面的过程,即对于一个叶子节点对应的集合,我们找出集合中所有叶子节点,那么这些叶子节点都应连向同一个非叶节点。
时间复杂度 \(\dfrac{n^3}{\omega}\)。
const int MAXN=1000;
int n;
bitset<MAXN+5> a[MAXN+5],b[MAXN+5],is;
int main(){
scanf("%d",&n);bool flg=1;
for(int i=1;i<=n;i++){
int len;scanf("%d",&len);flg&=(len==n);
while(len--){int x;scanf("%d",&x);a[i][x]=1;}
} if(flg){
for(int i=1;i<n;i++) printf("%d %d\n",i,n);
return 0;
} vector<pii> res;
for(int i=1;i<=n;i++) for(int j=1;j<=n;j++){
bitset<MAXN+5> tmp=a[i]&a[j];
if(tmp.count()==2){
int x=tmp._Find_first(),y=tmp._Find_next(x);
if(b[x][y]) continue;
b[x].set(y);b[y].set(x);is[x]=is[y]=1;
b[x][x]=b[y][y]=1;res.pb(mp(x,y));
}
} if(is.count()==2){
int r1=0,r2=0;
static int vis[MAXN+5];
for(int i=1;i<=n;i++) if(is[i]) (r1)?(r2=i):(r1=i);
for(int i=1;i<=n;i++) if(a[i].count()<n){
for(int j=1;j<=n;j++) if(!is[j]&&a[i][j]) vis[j]=1;
break;
}
for(int i=1;i<=n;i++) if(!is[i]){
if(vis[i]) res.pb(mp(r1,i));
else res.pb(mp(r2,i));
} for(pii e:res) printf("%d %d\n",e.fi,e.se);
} else {
static bool vis[MAXN+5];
for(int i=1;i<=n;i++){
bool flg=1;
for(int j=1;j<=n;j++) if((a[i]&a[j])==a[j]&&a[i]!=a[j])
flg=0;
if(!flg) continue;
bitset<MAXN+5> tmp=a[i]&is;
for(int j=1;j<=n;j++) if(tmp[j]&&b[j]==tmp){
if(!vis[j]){
for(int k=1;k<=n;k++) if(!is[k]&&a[i][k]) res.pb(mp(j,k));
vis[j]=1;
}
break;
}
} for(pii e:res) printf("%d %d\n",e.fi,e.se);
}
return 0;
}
/*
5
5 1 2 3 4 5
4 1 3 4 5
4 3 4 5 2
3 2 5 4
3 1 4 3
*/
Codeforces 566E - Restoring Map(bitset 优化构造)的更多相关文章
- Codeforces.566E.Restoring Map(构造)
题目链接 \(Description\) 对于一棵树,定义某个点的邻居集合为所有距离它不超过\(2\)的点的集合(包括它自己). 给定\(n\)及\(n\)个点的邻居集合,要求构造一棵\(n\)个点的 ...
- @codefoces - 566E@ Restoring Map
目录 @description@ @solution@ @accepted code@ @details@ @description@ 对于一棵 n 个点的树,我们称两个点是相邻的当且仅当两个点的距离 ...
- Codeforces Round #207 (Div. 1) D - Bags and Coins 构造 + bitset优化dp + 分段查找优化空间
D - Bags and Coins 思路:我们可以这样构造,最大的那个肯定是作为以一个树根,所以我们只要找到一个序列a1 + a2 + a3 .... + ak 并且ak为 所有点中最大的那个,那么 ...
- Codeforces 788C The Great Mixing(背包问题建模+bitset优化或BFS)
[题目链接] http://codeforces.com/problemset/problem/788/C [题目大意] 给出一些浓度的饮料,要求调出n/1000浓度的饮料,问最少需要多少升饮料 [题 ...
- Axel and Marston in Bitland CodeForces - 782F (bitset优化)
题目链接 $dp[0/1][i][x][y]$表示起始边为0/1, 走$2^i$ 步, 是否能从$x$走到$y$ 则有转移方程 $dp[z][i][x][y]\mid=dp[z][i-1][x][k] ...
- Codeforces Round #390 (Div. 2) E(bitset优化)
题意就是一个给出2个字符矩阵,然后进行匹配,输出每个位置的匹配的结果 (超出的部分循环处理) 一种做法是使用fft,比较难写,所以没有写 这里使用一个暴力的做法,考虑到一共只出现26个字符 所以使用一 ...
- Codeforces 576D - Flights for Regular Customers(bitset 优化广义矩阵乘法)
题面传送门 题意: 有一张 \(n\) 个点 \(m\) 条边的有向图,你初始在 \(1\) 号点,边上有边权 \(c_i\) 表示只有当你经过至少 \(c_i\) 条边的时候你才能经过第 \(i\) ...
- Codeforces 512E - Fox And Polygon(构造)
Codeforces 题面传送门 & 洛谷题面传送门 中规中矩的构造题一道. 首先考虑将两张图都向一个中间状态转化.方便起见我们取所有点都连向 \(1\) 号点的情形作为中间状态. 考虑怎样从 ...
- hdu 5745 La Vie en rose DP + bitset优化
http://acm.hdu.edu.cn/showproblem.php?pid=5745 这题好劲爆啊.dp容易想,但是要bitset优化,就想不到了. 先放一个tle的dp.复杂度O(n * m ...
随机推荐
- python打印爱心
print('\n'.join([''.join([('AndyLove'[(x-y)%8]if((x*0.05)**2+(y*0.1)**2-1)**3-(x*0.05)**2*(y*0.1)**3 ...
- JVM:Java中的引用
JVM:Java中的引用 本笔记是根据bilibili上 尚硅谷 的课程 Java大厂面试题第二季 而做的笔记 在原来的时候,我们谈到一个类的实例化 Person p = new Person() 在 ...
- JVM:类加载与字节码技术-2
JVM:类加载与字节码技术-2 说明:这是看了 bilibili 上 黑马程序员 的课程 JVM完整教程 后做的笔记 内容 这部分内容在上一篇笔记中: 类文件结构 字节码指令 编译期处理 类加载阶段 ...
- 使用vuex简单的实现系统中的状态管理
最近项目中用到了vue,其中状态的集中管理使用到了vuex,因此就学习vuex做一个简单的记录.vuex的官方网址如下: https://vuex.vuejs.org/zh-cn/ vuex 当我们 ...
- FastAPI 学习之路(二十九)使用(哈希)密码和 JWT Bearer 令牌的 OAuth2
既然我们已经有了所有的安全流程,就让我们来使用 JWT 令牌和安全哈希密码让应用程序真正地安全. 关于 JWT 它是一个将 JSON 对象编码为密集且没有空格的长字符串的标准.字符串看起来像这样: e ...
- (六)、Docker 之 Dockerfile
1.什么是Dockerfile Dockerfile是用来构建Docker镜像的构建文件,是由一系列命令和参数构成的脚本. 2.Dockerfile解析过程 前提认知: 每条保留字指令都必须为大写字母 ...
- gdal3.1.0+VS2017+geos+kml编译总结
1.简介 gdal3.1.0编译过程中必须依赖proj,编译gdal必须要编译proj,proj的编译需要sqlite3,因此想要编译gdal3.1.0需要先编译proj和sqlite3 2.关于sq ...
- 大神教零基础入门如何快速高效的学习c语言开发
零基础如果更快更好的入门C语言,如何在枯燥的学习中找到属于自己的兴趣,如果把学习当成一种事务性的那以后的学习将会很难有更深入的进步,如果带着乐趣来完成学习那将越学越有意思这样才会让你有想要更深入学习的 ...
- 零基础学习Linux心得总结
很多同学接触linux不多,对linux平台的开发更是一无所知. 而现在的趋势越来越表明,作为一个优秀的软件开发人员,或计算机it行业从业人员,="" 掌握linux是一种很重要的 ...
- C/C++中浮点数输出精度的问题
本文使用C++语言书写,对于C的小伙伴们,如果编译不通过的话--就说明C里面没有这个内容,可以跳过 通常来说,我们书写程序主要只用整形变量 (signed/unsigned) (long/long l ...