题目描述

输入输出格式

输入格式:

输入文件名:TRO.IN

输入文件仅有一行,不超过10000个字符,表示一个二叉树序列。

输出格式:

输出文件名:TRO.OUT

输出文件也只有一行,包含两个数,依次表示最多和最少有多少个点能够被染成绿色。

输入输出样例

输入样例#1:

1122002010
输出样例#1:

5 2

似乎已经很久没有写博客了
今天是noip2017初赛,离noip2017复赛还有不到一个月,我要加油做题了! 题解:
首先得能通过输入的一串树把树的形态搞出来。自己模拟模拟后发现这个“二叉树序列”跟dfs差不多,用一个dfs就知道树是什么样的了。
在dfs时记录该节点有几个子节点(只可能是0、1、2),顺便记下子节点的编号。
之后计算最多最少几个节点被染成绿色就可以用dp了
至于问什么会想到dp呢?我猜是一个经验问题吧。
一开始想起来虽觉得不好算,但大问题可以转化为子问题后解决。对于每一个小部分求出最值后进一步求出稍大一部分的最值。
由于要求最大值与最小值,所以我用了两个dp数组。
dp[i][k]表示第i号节点的颜色为k,它与它子树中颜色为绿的的最大值。(在前面的dfs中根据dfs序可知每一个子树中的节点编号连续,且根节点是最小的)
转移方程有些长,直接看下面代码吧,懒得再写一遍了。
方程基本就是对于该节点的一种状态求出子节点可能的不同状态(状态数并不多)中的最值。
之后就可以啦 总结一下思路:
弄清楚“二叉树序列”的本质 -> 发现大问题难以求解,但可通过子问题计算出 -> 进行dp 注意:题目很坑,数据范围是有问题的,最多可能输入长度为500000 代码:
 #include<cstdio>
#include<iostream>
using namespace std; const int MAXN=;
int sonnum[MAXN],son[MAXN][];
int num=,root; char ch;
void input(int u){
sonnum[u]=ch-'';
if(ch-''==) return;
else if(ch-''==){
son[u][]=++num;
ch=getchar();
input(num);
}
else{
son[u][]=++num;
ch=getchar();
input(num);
son[u][]=++num;
fa[num]=u;
ch=getchar();
input(num);
}
}
int dp[MAXN][],dp2[MAXN][]; int main()
{
int i,j,x;
ch=getchar();
while(ch<'' || ch>'') ch=getchar(); root=++num;
input(root); for(i=num;i>;i--){
if(sonnum[i]==){
dp[i][]=;
dp[i][]=dp[i][]=; dp2[i][]=;
dp2[i][]=dp2[i][]=;
}
else if(sonnum[i]==){
dp[i][]=max(dp[son[i][]][],dp[son[i][]][])+;
dp[i][]=max(dp[son[i][]][],dp[son[i][]][]);
dp[i][]=max(dp[son[i][]][],dp[son[i][]][]); dp2[i][]=min(dp2[son[i][]][],dp2[son[i][]][])+;
dp2[i][]=min(dp2[son[i][]][],dp2[son[i][]][]);
dp2[i][]=min(dp2[son[i][]][],dp2[son[i][]][]);
}
else {
dp[i][]=max(dp[son[i][]][]+dp[son[i][]][],dp[son[i][]][]+dp[son[i][]][])+;
dp[i][]=max(dp[son[i][]][]+dp[son[i][]][],dp[son[i][]][]+dp[son[i][]][]);
dp[i][]=max(dp[son[i][]][]+dp[son[i][]][],dp[son[i][]][]+dp[son[i][]][]); dp2[i][]=min(dp2[son[i][]][]+dp2[son[i][]][],dp2[son[i][]][]+dp2[son[i][]][])+;
dp2[i][]=min(dp2[son[i][]][]+dp2[son[i][]][],dp2[son[i][]][]+dp2[son[i][]][]);
dp2[i][]=min(dp2[son[i][]][]+dp2[son[i][]][],dp2[son[i][]][]+dp2[son[i][]][]);
}
}
printf("%d ",max(dp[][],max(dp[][],dp[][])));
printf("%d\n",min(dp2[][],min(dp2[][],dp2[][])));
return ;
}

洛谷P2585 [ZJOI2006]三色二叉树的更多相关文章

  1. 【树形DP】洛谷P2585 [ZJOI2006] 三色二叉树

    [树形DP]三色二叉树 标签(空格分隔): 树形DP [题目] 一棵二叉树可以按照如下规则表示成一个由0.1.2组成的字符序列,我们称之为"二叉树序列S": 0 该树没有子节点 1 ...

  2. 洛谷P2585 [ZJOI2006]三色二叉树(树形dp)

    传送门 设$dp[u][i]$表示点$u$颜色为$i$时最多(最少)的绿点个数(这里用$0$表示绿点) 然后直接用树形dp就可以了 记得把情况讨论清楚 //minamoto #include<b ...

  3. 【洛谷P2585】三色二叉树

    题目大意:给定一个二叉树,可以染红绿黄三种颜色,要求父节点和子节点的颜色不同,且如果一个节点有两个子节点,那么两个子节点之间的颜色也不同.求最多和最少有多少个节点会被染成绿色. 题解:加深了对二叉树的 ...

  4. 洛谷 2585 [ZJOI2006]三色二叉树——树形dp

    题目:https://www.luogu.org/problemnew/show/P2585 可以把不是绿色的记成一种.仔细一想不会有冲突.如果自己是绿色,孩子的不同颜色不会冲突:如果自己不是绿色,自 ...

  5. luogu P2585 [ZJOI2006]三色二叉树

    P2585 [ZJOI2006]三色二叉树 题目描述 输入输出格式 输入格式: 输入文件名:TRO.IN 输入文件仅有一行,不超过10000个字符,表示一个二叉树序列. 输出格式: 输出文件名:TRO ...

  6. P2585 [ZJOI2006]三色二叉树

    题目描述 输入输出格式 输入格式: 输入文件名:TRO.IN 输入文件仅有一行,不超过500000个字符,表示一个二叉树序列. 输出格式: 输出文件名:TRO.OUT 输出文件也只有一行,包含两个数, ...

  7. BZOJ1864[ZJOI2006]三色二叉树[树形DP]

    1864: [Zjoi2006]三色二叉树 Time Limit: 1 Sec  Memory Limit: 64 MBSubmit: 773  Solved: 548[Submit][Status] ...

  8. 【BZOJ1864】[Zjoi2006]三色二叉树 树形DP

    1864: [Zjoi2006]三色二叉树 Description Input 仅有一行,不超过500000个字符,表示一个二叉树序列. Output 输出文件也只有一行,包含两个数,依次表示最多和最 ...

  9. BZOJ 1864: [Zjoi2006]三色二叉树( 树形dp )

    难得的ZJOI水题...DFS一遍就行了... ----------------------------------------------------------------------- #inc ...

随机推荐

  1. Java 学习笔记(11)——异常处理

    异常是程序中的一些错误,但并不是所有的错误都是异常,并且错误有时候是可以避免的. 比如说,你的代码少了一个分号,那么运行出来结果是提示是错误 java.lang.Error:如果你用System.ou ...

  2. PHP常用函数拾遗

     PHP中常用过滤函数addslashes().mysql_real_escape_string().mysql_escape_string() 如addslashes().mysql_real_es ...

  3. Visioi形状相关应用

    选择手柄为白点  按住shift的同时移动白点更为灵活 黄色的点就是控制手柄(只有一维图形有) 当调整形状出现绿色边的时候说明:这个时候这个形状的边等于了某个形状的长 铅笔工具可以移动控制点来更形状 ...

  4. TestStand 界面重置【小技巧】

    有几种情况可能会使用到这个功能: (1)当界面调整的很乱的时候 (2)当界面突然消失的时候(但是软件进程还在)--快捷键 Alt+V 会弹出菜单,再点击Reset UI Configuration即可 ...

  5. 图解Go里面的sync.Map了解编程语言核心实现源码

    基础筑基 在大多数语言中原始map都不是一个线程安全的数据结构,那如果要在多个线程或者goroutine中对线程进行更改就需要加锁,除了加1个大锁,不同的语言还有不同的优化方式, 像在java和go这 ...

  6. Omnigraffle 许可证

    名字:Appked 序列号:MFWG-GHEB-HYTW-CGHT-CSXU-QCNC-SXU

  7. [Ljava.lang.String;@3e5084c9:是一个字符串数组的字节码表示

    [Ljava.lang.String;@3e5084c9:是一个字符串数组的字节码表示 打印一个字符串数组的话,会发现 String[] arr = new String[10]; // String ...

  8. 010 Ceph RGW对象存储

    一.对象存储 1.1 介绍 通过对象存储,将数据存储为对象,每个对象除了包含数据,还包含数据自身的元数据 对象通过Object ID来检索,无法通过普通文件系统操作来直接访问对象,只能通过API来访问 ...

  9. $bzoj4237$稻草人 $cdq$分治

    正解:$cdq$分治 解题报告: 传送门$QwQ$ $umm$总感觉做过这题的亚子,,,? 先把坐标离散化,然后把所有点先按$x$排序$QwQ$,然后用类似平面最近点对的方法,先分别解决$mid$两侧 ...

  10. 【一起学源码-微服务】Nexflix Eureka 源码十三:Eureka源码解读完结撒花篇~!

    前言 想说的话 [一起学源码-微服务-Netflix Eureka]专栏到这里就已经全部结束了. 实话实说,从最开始Eureka Server和Eureka Client初始化的流程还是一脸闷逼,到现 ...