poj 3345 树形DP 附属关系+输入输出(好题)
题目连接:http://acm.hust.edu.cn/vjudge/problem/17665
参考资料:http://blog.csdn.net/woshi250hua/article/details/7684771
题目大意:xx大佬要竞选xx职位,现一共有n个国家,获得xx职位至少需要m个国家的支持,某个国家下面会有若干个附属国家,这个代表获得这个国家的支持就可以获得一群国家的支持。想要获得这个国家的支持,就必须拿钻石去贿赂,好厚黑。问获得xx职位最少需要多少钻石。
这道题有几个要点:
这里不是一棵树,而是一个森林。所以要记录所有根节点,并添加一个主根
for(i=;i<=n;i++) if(!vis2[i]) add(,i);
选了根节点就等同于选了所有的子节点,为了实现这个需要注意两个地方
先看代码
void dfs(int rt)
{
vis[rt]=;
num[rt]=dp[rt][]=;
for(int i = head[rt];i!=-;i=tree[i].next)
{
int y = tree[i].y;
if(vis[y]) continue;
dfs(y);
num[rt]+=num[y];
for(int j=num[rt];j>=;j--)
{
for(int k=;k<=j;k++)
{
dp[rt][j] = min(dp[rt][j],dp[rt][j-k]+dp[y][k]);
}
}
}
num[rt]++;
dp[rt][num[rt]] = cost[rt];
}
- num[rt]一开始初始化为0,到了最后才+1,这是因为同时选根节点和子节点已经没有意义了,所以对于一个根只需要遍历子节点的情况,并在最后将选了根节点的情况加入即可。
- 在第二个循环里,如果选子必选根,则K<J;如果选子和选根没关系,则K<=J,即根可以为0
输入:
这里一开始输入两个数字,却是用#结束,怎么办?
两种思路:1.字符串转数字。2.用sscanf。
输入没有给定数量,通过换行结束怎么办?
用getchar()判断换行符号,不是换行符号就接着输入
其他的和一般的树形DP没什么区别,完整代码如下:
#include <iostream>
#include <string>
#include <cstring>
#include <cstdlib>
#include <cstdio>
#include <cmath>
#include <algorithm>
#include <stack>
#include <queue>
#include <cctype>
#include <vector>
#include <iterator>
#include <set>
#include <map>
#include <sstream>
using namespace std; #define mem(a,b) memset(a,b,sizeof(a))
#define pf printf
#define sf scanf
#define spf sprintf
#define pb push_back
#define debug printf("!\n")
#define MAXN 200+5
#define MAX(a,b) a>b?a:b
#define blank pf("\n")
#define LL long long
#define ALL(x) x.begin(),x.end()
#define INS(x) inserter(x,x.begin())
#define pqueue priority_queue
#define INF 0x3f3f3f3f int n,m; struct node{int y,val,next;}tree[MAXN<<]; int head[MAXN],vis[MAXN],ptr=,val[MAXN],dp[MAXN][MAXN]; int cost[MAXN],vis2[MAXN],num[MAXN]; map<string,int> mmap; void init()
{
mem(head,-);
mem(vis,);
mem(dp,INF);
mem(vis2,);
ptr=;
mmap.clear();
}
void add(int x,int y)
{
tree[ptr].y = y;
tree[ptr].next = head[x];
head[x] = ptr++;
} void dfs(int rt)
{
vis[rt]=;
num[rt]=dp[rt][]=;
for(int i = head[rt];i!=-;i=tree[i].next)
{
int y = tree[i].y;
if(vis[y]) continue;
dfs(y);
num[rt]+=num[y];
for(int j=num[rt];j>=;j--)
{
for(int k=;k<=j;k++)
{
dp[rt][j] = min(dp[rt][j],dp[rt][j-k]+dp[y][k]);
}
}
}
num[rt]++;
dp[rt][num[rt]] = cost[rt];
} int main()
{
int i,j,k;
char tmp[],a[],b[];
while(gets(tmp) && tmp[]!='#')
{
sscanf(tmp,"%d%d",&n,&m);
init();
int tot=,ta,tb,ans=INF;
for(i=;i<=n;i++)
{
sf("%s%d",a,&k);
if(mmap.find(string(a))==mmap.end())
mmap[string(a)] = tot++;
ta = mmap[string(a)];
cost[ta] = k; while(getchar()!='\n')
{
sf("%s",b);
if(mmap.find(string(b))==mmap.end())
mmap[string(b)] = tot++;
tb = mmap[string(b)];
add(ta,tb);
add(tb,ta);
vis2[tb] = ;
}
}
for(i=;i<=n;i++)
{
if(!vis2[i]) add(,i);
}
dfs();
for(i=m;i<=n;i++) ans = min(ans,dp[][i]);
pf("%d\n",ans);
}
}
poj 3345 树形DP 附属关系+输入输出(好题)的更多相关文章
- Fire (poj 2152 树形dp)
Fire (poj 2152 树形dp) 给定一棵n个结点的树(1<n<=1000).现在要选择某些点,使得整棵树都被覆盖到.当选择第i个点的时候,可以覆盖和它距离在d[i]之内的结点,同 ...
- poj 1463(树形dp)
题目链接:http://poj.org/problem?id=1463 思路:简单树形dp,如果不选父亲节点,则他的所有的儿子节点都必须选,如果选择了父亲节点,则儿子节点可选,可不选,取较小者. #i ...
- poj 2486( 树形dp)
题目链接:http://poj.org/problem?id=2486 思路:经典的树形dp,想了好久的状态转移.dp[i][j][0]表示从i出发走了j步最后没有回到i,dp[i][j][1]表示从 ...
- poj 3140(树形dp)
题目链接:http://poj.org/problem?id=3140 思路:简单树形dp题,dp[u]表示以u为根的子树的人数和. #include<iostream> #include ...
- POJ 2342 树形DP入门题
有一个大学的庆典晚会,想邀请一些在大学任职的人来參加,每一个人有自己的搞笑值,可是如今遇到一个问题就是假设两个人之间有直接的上下级关系,那么他们中仅仅能有一个来參加,求请来一部分人之后,搞笑值的最大是 ...
- Strategic game(POJ 1463 树形DP)
Strategic game Time Limit: 2000MS Memory Limit: 10000K Total Submissions: 7490 Accepted: 3483 De ...
- POJ 1155 树形DP
题意:电视台发送信号给很多用户,每个用户有愿意出的钱,电视台经过的路线都有一定费用,求电视台不损失的情况下最多给多少用户发送信号. 转自:http://www.cnblogs.com/andre050 ...
- POJ 3342 树形DP+Hash
这是很久很久以前做的一道题,可惜当时WA了一页以后放弃了. 今天我又重新捡了起来.(哈哈1A了) 题意: 没有上司的舞会+判重 思路: hash一下+树形DP 题目中给的人名hash到数字,再进行运算 ...
- POJ 1770 树形DP
咋一看确实想到的是树形DP,但是我一开始也马上想到环的情况,这样应该是不可以进行树形DP的,然后我自以为是地想用有向图代替无向图,而且总是从能量高的指向能量低的,这样自以为消除了环,但是其实是不对滴, ...
随机推荐
- webpack4 入门(一)
一.简介 WebPack可以看做是模块打包机:它做的事情是,分析你的项目结构,找到JavaScript模块以及其它的一些浏览器不能直接运行的拓展语言(Scss,less, TypeScript等),并 ...
- [CQOI2012][bzoj2668] 交换棋子 [费用流]
题面 传送门 思路 抖机灵 一开始看到这题我以为是棋盘模型-_-|| 然而现实是骨感的 后来我尝试使用插头dp来交换,然后又惨死 最后我不得不把目光转向那个总能化腐朽为神奇的算法:网络流 思维 我们要 ...
- 对于vue的一些理解
首先是组件之间的通信方式 父组件到子组件的通信: props和$refs 子组件到父组件的通信: events 和 $parents 以及 $root 组件之间的通信 eventBus和vuex ev ...
- J.U.C-其他组件
J.U.C-其他组件 FutureTask 介绍Callable时我们知道它可以有返回值,返回值通过Future进行封装.FutrueTask实现了RunnableFuture接口,该接口继承了R ...
- Hibernate上传数据到数据库,从数据库读取数据到本地模板代码
1.Hibernate上传数据到数据库: //创建一个session对象 Session session1=HibernateTools.getSession(); //Fenciresult数据库表 ...
- BeautifulSoup4模块的使用
1. 安装 pip3 install beautifulsoup42. 使用 from bs4 import BeautifulSoup obj = BeautifulSoup("HTML内 ...
- python3 zip()函数笔记
a=[1,2,3]b=[4,5,6] for A ,B in zip(a,b):#用zip()函数整体打包 print(A,B)
- mpdf Could not find image file (http://local.com/xxxxx)
记录一下昨天和今天遇到的,yii2使用mpdf的时候,图片是使用php方法生成的二维码,所以图片地址为http://local.com/xxxxx,url中携带不同的参数. 但是开启了 $mpdf-& ...
- 操作集合的线程安全考虑——java
运行场景:多个线程同时调用ArrayList存放元素 两个线程A和B,在A线程调用的时候,list中暂时还未有元素存在,此时,list的size值为0,同时A在添加元素的时候,add进了一个元素,此时 ...
- 树莓派安装Raspbian系统以及相关配置(通过Windows)
1.准备 树莓派3B+(E14) 一张内存卡 (至少16G) 一个读卡器 普通电脑显示器,键盘,鼠标 2.操作系统烧制(下面的操作都是在Windows中操作,通过读开启读取内存卡) Raspbian系 ...