洛谷题目页面传送门 & CodeForces题目页面传送门

给定一棵树\(T=(V,E),|V|=2^n-2,|E|=2^n-3\),输出所有的\(x\),使得存在一棵满二叉树\(T'\),将\(T'\)中节点\(x\)的一个儿子删除并把这个儿子的所有儿子接到\(x\)下后等于\(T\)。升序输出。

\(n\le17\)。

题目没有说以哪个点为根,也就是每个点都有可能是根,很自然地想到可以二次扫描与换根。先考虑选一个点作为根,那显然满足条件的改补的节点的父结点最多有\(1\)个。这个父结点可以DP出来。

我们将一个子树分类讨论:

  1. 是一棵满二叉树。设它的深度为\(d\),则记这颗子树的特征为有序对\((0,d)\)。这种情况发生当且仅当它有\(2\)棵子树并且都是矮\(1\)层的满二叉树。特殊地,如果它的大小为\(1\),则它的特征为\((0,1)\);

  2. 还原一个节点之后为满二叉树。设还原之后的深度为\(d\),补的节点的父结点为\(x\),则记这棵子树的特征为有序对\((x,d)\)。这种情况发生当且仅当以下任意一个条件为真:

    1. 它的根为\(x\),有\(1\)棵子树并且这棵子树大小为\(1\),此时应将改补的节点直接补在\(x\)下;
    2. 它的根为\(x\),有\(3\)棵子树并且其中\(1\)棵为矮\(1\)层的满二叉树,另\(2\)棵为矮\(2\)层的满二叉树,此时应将改补的节点补在\(x\)下并将\(2\)棵矮\(2\)层的字树接在改补的节点下;
    3. 它有\(2\)棵子树并且一棵为矮\(1\)层的满二叉树,另一颗补一个父结点为\(x\)的节点之后为矮\(1\)层的满二叉树;
  3. 不管补不补节点都不能成为满二叉树。记它的特征为有序对\((-1,-1)\)。显然,不满足\(1,2\)则为此种情况。

设\(dp_i\)为以\(1\)为根时以\(i\)为根的子树的特征,则状态转移方程是(太♂难写已隐藏)。这样一遍\(\mathrm O(2^n)\)DFS则可求出所有节点的DP值。而我们希望找到所有节点为根时的根节点DP值,这个可以二次扫描与换根,即再一遍DFS。每到达一个节点\(x\)时,目前所有节点的DP值均是以\(x\)为整棵树的根的,所以若\(dp_x=(y,n)(y>0)\),就将\(y\)加入答案序列。那么此时若要将它的某个儿子\(s\)改为根,那么改变的只有\(dp_x\)和\(dp_s\)。我们可以改一下它们的儿子集合(涉及添加和删除,用set较为方便),重新算DP值,然后再DFS到\(s\),此时整棵树的根为\(s\)了。从\(s\)回溯时,再还原\(x\)和\(s\)的儿子集合和DP值,去找别的儿子即可。由于换根操作只需要改变\(2\)个节点的信息,所以复杂度是有保证的,一共\(\mathrm O(2^n\log2^n)=\mathrm O(2^nn)\)(\(\log\)是set)。

下面贴代码:

#include<bits/stdc++.h>
using namespace std;
#define pb push_back
#define mp make_pair
#define X first
#define Y second
const int N=17;
int n;
vector<int> nei[1<<N];//邻接表
set<int> son[1<<N];//儿子集合
void dfs(int x=1,int fa=0){//求出所有节点的儿子集合
for(int i=0;i<nei[x].size();i++){
int y=nei[x][i];
if(y==fa)continue;
son[x].insert(y);
dfs(y,x);
}
}
pair<int,int> f[1<<N];//DP值,即以[1]为根的子树的特征
void calc_f(int x){//通过儿子集合计算DP值,即那个难写的状态转移方程
if(son[x].size()==0)f[x]=mp(0,1);
else if(son[x].size()==1)f[x]=f[*son[x].begin()]==mp(0,1)?mp(x,2):mp(-1,-1);
else if(son[x].size()==2){
pair<int,int> x1=f[*son[x].begin()],x2=f[*++son[x].begin()];
if(x1>x2)swap(x1,x2);
if(!x1.X&&!x2.X)f[x]=x1.Y==x2.Y?mp(0,x1.Y+1):mp(-1,-1);
else if(!x1.X&&x2.X>0)f[x]=x1.Y==x2.Y?mp(x2.X,x1.Y+1):mp(-1,-1);
else f[x]=mp(-1,-1);
}
else if(son[x].size()==3){
pair<int,int> x1=f[*son[x].begin()],x2=f[*++son[x].begin()],x3=f[*++ ++son[x].begin()];
if(x1>x2)swap(x1,x2);if(x2>x3)swap(x2,x3);if(x1>x2)swap(x1,x2);
if(!x1.X&&!x2.X&&!x3.X)f[x]=x1.Y==x2.Y&&x2.Y+1==x3.Y?mp(x,x3.Y+1):mp(-1,-1);
else f[x]=mp(-1,-1);
}
else f[x]=mp(-1,-1);
// printf("f[%d]=(%d,%d)\n",x,f[x].X,f[x].Y);
}
void dp(int x=1,int fa=0){//一遍DFS求出以1为整棵树的根时的DP数组
for(int i=0;i<nei[x].size();i++){
int y=nei[x][i];
if(y==fa)continue;
dp(y,x);
}
calc_f(x);
}
vector<int> ans;//答案序列
void dfs0(int x=1,int fa=0){//二次扫描
if(f[x].X>0)ans.pb(f[x].X);//加入答案序列
for(int i=0;i<nei[x].size();i++){
int y=nei[x][i];
if(y==fa)continue;
son[x].erase(y);son[y].insert(x);calc_f(x);calc_f(y);//改变儿子集合,重新算DP值
dfs0(y,x);
son[x].insert(y);son[y].erase(x);calc_f(y);calc_f(x);//还原
}
}
int main(){
cin>>n;
for(int i=1;i<=(1<<n)-3;i++){
int x,y;
cin>>x>>y;
nei[x].pb(y);nei[y].pb(x);
}
dfs();
dp();
dfs0();
cout<<ans.size()<<"\n";
sort(ans.begin(),ans.end());
for(int i=0;i<ans.size();i++)cout<<ans[i]<<" ";
return 0;
}

CodeForces 1228F One Node is Gone的更多相关文章

  1. babeljs源码

    babel.min.js!function(e,t){"object"==typeof exports&&"object"==typeof mo ...

  2. node搜索codeforces 3A - Shortest path of the king

    发一下牢骚和主题无关: 搜索,最短路都可以     每日一道理 人生是洁白的画纸,我们每个人就是手握各色笔的画师:人生也是一条看不到尽头的长路,我们每个人则是人生道路的远足者:人生还像是一块神奇的土地 ...

  3. CodeForces - 274B Zero Tree

    http://codeforces.com/problemset/problem/274/B 题目大意: 给定你一颗树,每个点上有权值. 现在你每次取出这颗树的一颗子树(即点集和边集均是原图的子集的连 ...

  4. Codeforces 划水

    Codeforces 566F 题目大意:给定$N$个数,任意两个数之间若存在一个数为另一个数的因数,那么这两个数存在边,求图中最大团. 分析:求一个图最大团为NP-Hard问题,一般不采用硬方法算. ...

  5. Codeforces Round #354 (Div. 2) ABCD

    Codeforces Round #354 (Div. 2) Problems     # Name     A Nicholas and Permutation standard input/out ...

  6. CodeForces 548D 单调栈

    Mike and Feet Time Limit:1000MS     Memory Limit:262144KB     64bit IO Format:%I64d & %I64u Subm ...

  7. HDU 3333 | Codeforces 703D 树状数组、离散化

    HDU 3333:http://acm.hdu.edu.cn/showproblem.php?pid=3333 这两个题是类似的,都是离线处理查询,对每次查询的区间的右端点进行排序.这里我们需要离散化 ...

  8. Codeforces VK CUP 2015 D. Closest Equals(线段树+扫描线)

    题目链接:http://codeforces.com/contest/522/problem/D 题目大意:  给你一个长度为n的序列,然后有m次查询,每次查询输入一个区间[li,lj],对于每一个查 ...

  9. Codeforces Round #363 (Div. 2)

    A题 http://codeforces.com/problemset/problem/699/A 非常的水,两个相向而行,且间距最小的点,搜一遍就是答案了. #include <cstdio& ...

随机推荐

  1. V4L2 API详解 Buffer的准备和数据读取

    1. 初始化 Memory Mapping 或 User Pointer I/O. int ioctl(int fd, int requestbuf, struct v4l2_requestbuffe ...

  2. 葫芦娃团队对火鸡堂、基于云的胜利冲锋队团队的Beta产品测试报告

    Beta项目互测 课程名称:软件工程1916|W(福州大学) 作业要求:Beta阶段团队项目互评 团队名称:葫芦娃队 作业目标:Beta项目互测 一.火鸡堂团队产品测试 1.截图(推荐用动态gif图录 ...

  3. 访问控制 RAM(Resource Access Management),知识点

    资料 网址 什么是访问控制 https://help.aliyun.com/document_detail/28672.html?spm=5176.11065259.1996646101.search ...

  4. Logstash 学习资料

    学习资料 网址 Logstash Reference(官方) https://www.elastic.co/guide/en/logstash/current/introduction.html

  5. Python学习进阶之薄弱点总结

    ''' 1.实现用户传入一个普通字符串, 返回字符串的md5加密结果的函数 ''' # import hashlib # # def M(str): # m = hashlib.md5() # m.u ...

  6. Linux 换 jdk 版本 环境没有生效

    Linux 换 jdk 版本 环境没有生效 把 jdk 1.7 换成 1.8, 路径设置好了后 用了下面两个都没有生效 . /etc/profile source ~/.bashrc 还是 jdk 1 ...

  7. selenium 动态元素的定位

    对于有些元素每次点击都是动态的  即每次都是不一样的  对于这种元素我们可以采用与他相关的其他静态的元素定位 比如 iframe 这个元素的id是动态的 每次都在变化 第一种就是我们观察下 是整个都是 ...

  8. 【java异常】【redis】ERR Client sent AUTH, but no password is set

    项目中使用jedis或redisson连接redis时,如果redis没有密码,但在配置文件中写为 spring: redis: database: 0 host: 127.0.0.1 passwor ...

  9. NOIP2013-2014提高组题目浅析

    1.前言 迎接NOIP的到来...在这段闲暇时间,决定刷刷水题.这里只是作非常简单的一些总结. 2.NOIP2014 <1> 生活大爆炸之石头剪刀布(模拟) 这是一道考你会不会编程的题目. ...

  10. luoguP2039 [AHOI2009]跳棋 巧妙的dp

    设\(f[i]\)表示在第\(i\)个格子上弄一个棋子的最小代价,前后扫两遍dp后统计答案即可. 代码 #include<bits/stdc++.h> using namespace st ...