吐槽:这道题真心坑...做了一整天,我太蒻了...

题意

构造一棵 $ n $ 个节点的树,要求满足以下条件:

  • 每个非叶子节点至少包含2个儿子;
  • 以节点 $ i $ 为根的子树中必须包含 $ c_i $ 个节点。

给出 $ n (n<=24)$ 和 $ c_i $ ,问是否存在符合条件的树。

分析

看到数据范围,第一时间想到的方法应该是状压DP,用数 $ s $ 表示已经被选入子树的点的集合,将两个以上小的子树合并为大的子树,看最后能否得到 $ s=1<<(n-1)-1 $ 这一集合。这么做的话,你就会…… 爆炸般地TLE

为什么?稍加分析就会发现,一共有 $ 2^n-1 $ 种状态,超过了 $ 1600 $ 万,其中绝大多数都是无法合并(有交集)的,遍历的时间大大超过了转移的时间。并且,题目问的是树是否存在,并不需要求出每一个状态,我们做了非常多的冗余操作。事实上,我们只需要找到一组解就可以了。那么,怎么快速确定某棵子树是否存在呢?

深搜+贪心。

我们发现,在确定比较大的子树时,我们总是需要用到小的子树,而大子树的存在性只与小子树的大小有关,与组合方式无关,所以我们可以dfs搜索子树的组合,并且一搜到解就退出。那么按照什么样的顺序搜索组合呢?可以直观地想到,在一开始放入的子树越大,越容易出解,所以我们在搜索每个子树的组合时按照从大到小的顺序搜,搜到解后删除原来的子树集合并用新的大子树代替以防止重复搜索,最后判断 $ 111...1 $ 这一集合是否存在。

另外,在读入 $ c_i $ 后按照大小排序,可以降低编程复杂度。

代码

AC,15ms,CF评测机真乃神机也

//by sclbgw7
#include <cstdio>
#include <cstring>
#include <algorithm>
#include <vector>
#define R register
using namespace std;
int si[25];//保存子树大小
vector<int> dp[25];//dp[i]存放表示大小为i的子树的集合 int dfs(int s,int x,int y)
{
if(x==0)
{
if(!b[s])
dp[si[y]].push_back(s),b[s]=1;
return 1;
}
if(s)//如果已经选择了一些小子树
{
for(int i=x;i>=1;--i)
for(int j=dp[i].size()-1;j>=0;--j)
if(!(s&dp[i][j]))
if(dfs(s|dp[i][j],x-i,y))
{dp[i].pop_back();return 1;}
return 0;
}
s=(1<<(y-1));
for(int i=x-2;i>=1;--i)//内部节点必须有两个以上的儿子
for(int j=dp[i].size()-1;j>=0;--j)
if(!(s&dp[i][j]))
if(dfs(s|dp[i][j],x-i-1,y))
{dp[i].pop_back();return 1;}
return 0;
} int main()
{
int n,m;
scanf("%d",&n);
for(int i=1;i<=n;++i)
scanf("%d",&si[i]);
sort(si+1,si+1+n);
int i;
for(i=1;si[i]==1;++i)
dp[1].push_back(1<<(i-1));
for(;i<=n;++i)
dfs(0,si[i],i);
if(dp[n].size())
printf("YES\n");
else
printf("NO\n");
return 0;
}

Codeforces 429C Guess the Tree(状压DP+贪心)的更多相关文章

  1. [多校联考2019(Round 5 T1)] [ATCoder3912]Xor Tree(状压dp)

    [多校联考2019(Round 5)] [ATCoder3912]Xor Tree(状压dp) 题面 给出一棵n个点的树,每条边有边权v,每次操作选中两个点,将这两个点之间的路径上的边权全部异或某个值 ...

  2. Codeforces Round #363 LRU(概率 状压DP)

    状压DP: 先不考虑数量k, dp[i]表示状态为i的概率,状态转移方程为dp[i | (1 << j)] += dp[i],最后考虑k, 状态表示中1的数量为k的表示可行解. #incl ...

  3. codeforces 8C. Looking for Order 状压dp

    题目链接 给n个物品的坐标, 和一个包裹的位置, 包裹不能移动. 每次最多可以拿两个物品, 然后将它们放到包里, 求将所有物品放到包里所需走的最小路程. 直接状压dp就好了. #include < ...

  4. Codeforces 895C Square Subsets(状压DP 或 异或线性基)

    题目链接  Square Subsets 这是白书原题啊 先考虑状压DP的做法 $2$到$70$总共$19$个质数,所以考虑状态压缩. 因为数据范围是$70$,那么我们统计出$2$到$70$的每个数的 ...

  5. 【题解】codeforces 8c Looking for Order 状压dp

    题目描述 Lena喜欢秩序井然的生活.一天,她要去上大学了.突然,她发现整个房间乱糟糟的--她的手提包里的物品都散落在了地上.她想把所有的物品都放回她的手提包.但是,这里有一点问题:她一次最多只能拿两 ...

  6. 刷题总结——树有几多愁(51nod1673 虚树+状压dp+贪心)

    题目: lyk有一棵树,它想给这棵树重标号. 重标号后,这棵树的所有叶子节点的值为它到根的路径上的编号最小的点的编号. 这棵树的烦恼值为所有叶子节点的值的乘积. lyk想让这棵树的烦恼值最大,你只需输 ...

  7. Codeforces 895C Square Subsets:状压dp【组合数结论】

    题目链接:http://codeforces.com/problemset/problem/895/C 题意: 给你n个数a[i].(n <= 10^5, 1 <= a[i] <= ...

  8. CodeForces 599E Sandy and Nuts 状压DP

    题意: 有一棵\(n(1 \leq n \leq 13)\)个节点的树,节点的标号为\(1 \sim n\),它的根节点是\(1\). 现在已知它的\(m(0 \leq m < n)\)条边,和 ...

  9. 【9.7校内测试】【二分+spfa】【最长上升子序列】【状压DP+贪心(?)】

    刘汝佳蓝书上的题,标程做法是从终点倒着$spfa$,我是二分答案正着$spfa$判断可不可行.效果是一样的. [注意]多组数据建边一定要清零啊QAQ!!! #include<iostream&g ...

随机推荐

  1. IOS取消performSelector警告

    #pragma clang diagnostic push #pragma clang diagnostic ignored "-Warc-performSelector-leaks&quo ...

  2. 视音频数据处理入门:FLV封装格式解析

    ===================================================== 视音频数据处理入门系列文章: 视音频数据处理入门:RGB.YUV像素数据处理 视音频数据处理 ...

  3. git 第一次关联远程仓库

    1.首先需要先git pull origin master 2.然后合并两个无关的仓库 git pull origin master --allow-unrelated-histories

  4. CSS3实战之content

    为元素添加内容 content属性术语内容生成和替换模块,该属性能够为指定元素添加内容. 取值如下 normal:默认值 string:文本内容 attr():插入元素的属性值 uri():插入一个外 ...

  5. javascript的未知尺寸图片保持比例水平垂直居中函数

    JavaScript的图片在容器内水平垂直居中的函数,利用图片加载获取图片大小,使之在父节点内水平垂直居中 展示方式有两种: 1.当参数keepImageFull为true:保持图片比例,使图片可完整 ...

  6. Djangoform表单Ajax控制跳转

    需求: 1:在登陆页面输入账号密码后,ajax异步提交数据给后端验证. 2:验证通过后,后端指定跳转页面,并把页面封装进返回的Json数据中,由ajax控制from表单跳转到目标页面 一:登陆页面HT ...

  7. 如何开启一个Django项目

    一:新建的Django工程 新建了一个Django工程后,工程会自动创建有两个templates文件夹和unitled文件夹,再加上一个manage.py文件. 二:Django开发的一般流程 在工程 ...

  8. ROS多线程订阅消息

    对于一些只订阅一个话题的简单节点来说,我们使用ros::spin()进入接收循环,每当有订阅的话题发布时,进入回调函数接收和处理消息数据.但是更多的时候,一个节点往往要接收和处理不同来源的数据,并且这 ...

  9. 解决 Electron 包下载太慢问题

    项目下新建 .npmrc 文件,加入如下配置: electron_mirror=https://npm.taobao.org/mirrors/electron/ 即使用淘宝的源,重新 npm inst ...

  10. 2016.5.16——leetcode:Reverse Bits(超详细讲解)

    leetcode:Reverse Bits 本题目收获 移位(<<  >>), 或(|),与(&)计算的妙用 题目: Reverse bits of a given 3 ...