Time Limit: 1000 mSec    Memory Limit : 32768 KB

 Problem Description

一棵二叉树可以按照如下规则表示成一个由0、1、2组成的字符序列,我们称之为“二叉树序列S”:

例如,下图所表示的二叉树可以用二叉树序列S=21200110来表示。

你的任务是要对一棵二叉树的节点进行染色。每个节点可以被染成红色、绿色或蓝色。并且,一个节点与其子节点的颜色必须不同,如果该节点有两个子节点,那么这两个子节点的颜色也必须不相同。给定一棵二叉树的二叉树序列,请求出这棵树中最多和最少有多少个点能够被染成绿色。

 Input

输入数据由多组数据组成。
每组数据仅有一行,不超过10000个字符,表示一个二叉树序列。

 Output

对于每组输入数据,输出仅一行包含两个整数,依次表示最多和最少有多少个点能够被染成绿色。

 Sample Input

1122002010

 Sample Output

5 2

【思路】先建树,然后进行树形DP;

#include<iostream>
#include<stdio.h>
#include<string.h>
using namespace std;
const int inf=0x7777777;
const int colorn=;
enum color{green, red, blue, none};//enum 枚举名{ 枚举值表 };在枚举值表中应罗列出所有可用值。这些值也称为枚举元素。
struct node
{
node *left,*right;
int maxg[colorn],ming[colorn];
node()
{
left=right=NULL;
memset(maxg,,sizeof(int)*colorn);
memset(ming,,sizeof(int)*colorn);
}
};
typedef node *T;//指针类型定义:T等价于node *定义,T tree声明等价于node *a声明;
const int N=;
char *build(T &tree,char *a)//建树
{
if(a[]=='')
{
tree=new node();
return a+;
}
else if(a[]=='')
{
tree=new node();
return build(tree->left,a+);
}
else
{
tree=new node();
char *pos=build(tree->left,a+);
return build(tree->right,pos);
}
}
void TDP(T &tree)
{
if(tree==NULL) return ;
TDP(tree->left);
TDP(tree->right);
if(tree->left==NULL&&tree->right==NULL)
{
tree->maxg[green]=;tree->maxg[red]=tree->maxg[blue]=; tree->ming[green]=;tree->ming[red]=tree->ming[blue]=;
}
else if(tree->left!=NULL&&tree->right==NULL)
{
tree->maxg[green]=max(tree->left->maxg[red],tree->left->maxg[blue])+;
tree->maxg[red]=max(tree->left->maxg[blue],tree->left->maxg[green]);
tree->maxg[blue]=max(tree->left->maxg[green],tree->left->maxg[red]); tree->ming[green]=min(tree->left->ming[red],tree->left->ming[blue])+;
tree->ming[red]=min(tree->left->ming[blue],tree->left->ming[green]);
tree->ming[blue]=min(tree->left->ming[red],tree->left->ming[green]);
}
else if(tree->right!=NULL&&tree->left==NULL)
{
tree->maxg[green]=max(tree->right->maxg[red],tree->right->maxg[blue])+;
tree->maxg[red]=max(tree->right->maxg[blue],tree->right->maxg[green]);
tree->maxg[blue]=max(tree->right->maxg[green],tree->right->maxg[red]); tree->ming[green]=min(tree->right->ming[red],tree->right->ming[blue])+;
tree->ming[red]=min(tree->right->ming[blue],tree->right->ming[green]);
tree->ming[blue]=min(tree->right->ming[red],tree->right->ming[green]);
}
else
{
tree->maxg[green]=max(tree->right->maxg[red]+tree->left->maxg[blue],tree->right->maxg[blue]+tree->left->maxg[red])+;
tree->maxg[red]=max(tree->right->maxg[green]+tree->left->maxg[blue],tree->right->maxg[blue]+tree->left->maxg[green]);
tree->maxg[blue]=max(tree->right->maxg[green]+tree->left->maxg[red],tree->right->maxg[red]+tree->left->maxg[green]); tree->ming[green]=min(tree->right->ming[red]+tree->left->ming[blue],tree->right->ming[blue]+tree->left->ming[red])+;
tree->ming[red]=max(tree->right->ming[green]+tree->left->ming[blue],tree->right->ming[blue]+tree->left->ming[green]);
tree->ming[blue]=max(tree->right->ming[green]+tree->left->ming[red],tree->right->ming[red]+tree->left->ming[green]); }
}
void destroy(T &tree)
{
if(tree==NULL) return ;
destroy(tree->left);
destroy(tree->right);
delete tree;
}
int main()
{
char a[N];
while(~scanf("%s",a))
{
T tree=NULL;
build(tree,a); TDP(tree);
int maxx=-inf,minn=inf;
for(int i=;i<colorn;i++)
{
if(tree->maxg[i]>maxx) maxx=tree->maxg[i];
if(tree->ming[i]<minn) minn=tree->ming[i];
}
printf("%d %d\n",maxx,minn);
destroy(tree);
}
return ;
}

三色二叉树_树形DP的更多相关文章

  1. BZOJ_1864_[Zjoi2006]三色二叉树_树形DP

    BZOJ_1864_[Zjoi2006]三色二叉树_树形DP 题意: 分析:递归建树,然后DP,从子节点转移. 注意到红色和蓝色没有区别,因为我们可以将红蓝互换而方案是相同的.这样的话我们只需要知道当 ...

  2. 三色二叉树 ---伪树形dp

    题目描述 一棵二叉树可以按照如下规则表示成一个由0.1.2组成的字符序列,我们称之为"二叉树序列S": 0 该树没有子节点 1S1 该树有一个子节点,S1为其二叉树序列 1S1S2 ...

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

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

  4. BZOJ-1864-[Zjoi2006]三色二叉树(树形dp)

    Description Input 仅有一行,不超过500000个字符,表示一个二叉树序列. Output 输出文件也只有一行,包含两个数,依次表示最多和最少有多少个点能够被染成绿色. Sample ...

  5. 1864. [ZJOI2006]三色二叉树【树形DP】

    Description Input 仅有一行,不超过500000个字符,表示一个二叉树序列. Output 输出文件也只有一行,包含两个数,依次表示最多和最少有多少个点能够被染成绿色. Sample ...

  6. BZOJ 1864:[Zjoi2006]三色二叉树(树DP)

    三色二叉树 问题描述 输入 仅有一行,不超过500000个字符,表示一个二叉树序列. 输出 输出文件也只有一行,包含两个数,依次表示最多和最少有多少个点能够被染成绿色. 样例输入 1122002010 ...

  7. [bzoj1812][IOI2006]riv_多叉树转二叉树_树形dp

    riv bzoj-1812 IOI-2006 题目大意:给定一棵n个点树,要求在上面建立k个收集站.点有点权,边有边权,整棵树的代价是每个点的点权乘以它和它的最近的祖先收集站的距离积的和. 注释:$1 ...

  8. BZOJ 1864 三色二叉树 - 树型dp

    传送门 题目大意: 给一颗二叉树染色红绿蓝,父亲和儿子颜色必须不同,两个儿子颜色必须不同,问最多和最少能染多少个绿色的. 题目分析: 裸的树型dp:\(dp[u][col][type]\)表示u节点染 ...

  9. 【BZOJ-1864】三色二叉树 树形DP

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

随机推荐

  1. OC 实例变量(Instance Var)和成员变量(member var)区别

    摘要:  Objective-C  引入了“实例变量"的概念,但同时, 也经常出现 “成员变量”的声音. 到底什么是实例变量,什么是成员变量,二者的区别是什么呢? 今天查看apple 的官方 ...

  2. PHPExcel 学习笔记

    首先到phpexcel官网上下载最新的phpexcel类,下周解压缩一个classes文件夹,里面包含了PHPExcel.php和PHPExcel的文件夹,这个类文件和文件夹是我们需要的,把class ...

  3. poj 3468 A Simple Problem with Integers 线段树第一次 + 讲解

    A Simple Problem with Integers Description You have N integers, A1, A2, ... , AN. You need to deal w ...

  4. 编写一个小Servlet程序

    1.编写一个java类,此类继承HttpServlet 创建完工程(见上一篇随笔:使用eclipse创建在myeclipse中运行的web工程),在src中新建一个包,包名字叫servlet:再新建一 ...

  5. 网易云课堂 OCP

    数据库DBA任务: 管理数据库可用性 设计并创建数据库 管理物理结构 管理基于设计的存储 管理安全性 网络管理 备份与恢复 数据库调整与优化 关系型数据库(RDBMS) 多个表数据之间存在着关系 关系 ...

  6. VS设置背景色减缓眼睛疲劳

    工具--选项--字体和颜色--(纯文本)项背景色--自定义... 色调:85 饱和度:123 亮度:205 可自己微调 字体设为10.

  7. Hadoop c++开发

    假设你有上百G的数据,你要统计出这些数据中,含有某些你感兴趣的内容的数据的有多少条,你会怎么做?在硬件条件允许的情况下,用hadoop并行计算是一个不错的选择. 为了使本文得以清晰地说明,我们不妨假设 ...

  8. EAX、ECX、EDX、EBX寄存器的作用

    注意:在计算加法时,实在32位的累加器上进行,并注意类型之间的转换,数据的截取问题 一般寄存器:AX.BX.CX.DXAX:累积暂存器,BX:基底暂存器,CX:计数暂存器,DX:资料暂存器 索引暂存器 ...

  9. MySQL的简单查询

    1.普通查询 select * from info; #查询所有内容 select Code,Name from Info #查询某几列 2.条件查询 select * from Info where ...

  10. a various of context

    ContextWrapper.getApplicationContext():Return the context of the single, global Application object o ...