题目连接: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 附属关系+输入输出(好题)的更多相关文章

  1. Fire (poj 2152 树形dp)

    Fire (poj 2152 树形dp) 给定一棵n个结点的树(1<n<=1000).现在要选择某些点,使得整棵树都被覆盖到.当选择第i个点的时候,可以覆盖和它距离在d[i]之内的结点,同 ...

  2. poj 1463(树形dp)

    题目链接:http://poj.org/problem?id=1463 思路:简单树形dp,如果不选父亲节点,则他的所有的儿子节点都必须选,如果选择了父亲节点,则儿子节点可选,可不选,取较小者. #i ...

  3. poj 2486( 树形dp)

    题目链接:http://poj.org/problem?id=2486 思路:经典的树形dp,想了好久的状态转移.dp[i][j][0]表示从i出发走了j步最后没有回到i,dp[i][j][1]表示从 ...

  4. poj 3140(树形dp)

    题目链接:http://poj.org/problem?id=3140 思路:简单树形dp题,dp[u]表示以u为根的子树的人数和. #include<iostream> #include ...

  5. POJ 2342 树形DP入门题

    有一个大学的庆典晚会,想邀请一些在大学任职的人来參加,每一个人有自己的搞笑值,可是如今遇到一个问题就是假设两个人之间有直接的上下级关系,那么他们中仅仅能有一个来參加,求请来一部分人之后,搞笑值的最大是 ...

  6. Strategic game(POJ 1463 树形DP)

    Strategic game Time Limit: 2000MS   Memory Limit: 10000K Total Submissions: 7490   Accepted: 3483 De ...

  7. POJ 1155 树形DP

    题意:电视台发送信号给很多用户,每个用户有愿意出的钱,电视台经过的路线都有一定费用,求电视台不损失的情况下最多给多少用户发送信号. 转自:http://www.cnblogs.com/andre050 ...

  8. POJ 3342 树形DP+Hash

    这是很久很久以前做的一道题,可惜当时WA了一页以后放弃了. 今天我又重新捡了起来.(哈哈1A了) 题意: 没有上司的舞会+判重 思路: hash一下+树形DP 题目中给的人名hash到数字,再进行运算 ...

  9. POJ 1770 树形DP

    咋一看确实想到的是树形DP,但是我一开始也马上想到环的情况,这样应该是不可以进行树形DP的,然后我自以为是地想用有向图代替无向图,而且总是从能量高的指向能量低的,这样自以为消除了环,但是其实是不对滴, ...

随机推荐

  1. Build SSH for Development on Windows Subsystem for Linux

    It seems that Windows Subsystem for Linux (WSL) is getting much more mature than the time when it fi ...

  2. Quadratic Residues POJ - 1808 二次剩余定理

    \(\color{#0066ff}{题目链接 }\) link \(\color{#0066ff}{ 题解 }\) 结论题 \((\frac{a}{p})=a^{\frac{p-1}{2}}\mod ...

  3. 换根DP+树的直径【洛谷P3761】 [TJOI2017]城市

    P3761 [TJOI2017]城市 题目描述 从加里敦大学城市规划专业毕业的小明来到了一个地区城市规划局工作.这个地区一共有ri座城市,<-1条高速公路,保证了任意两运城市之间都可以通过高速公 ...

  4. pandas实例美国人口分析

  5. rest_framework 的验证,权限,频率

    回到顶部 快速实例 Quickstart 回到顶部 序列化 创建一个序列化类 简单使用 开发我们的Web API的第一件事是为我们的Web API提供一种将代码片段实例序列化和反序列化为诸如json之 ...

  6. Java的Protected

    没想到接触Java这么多年,今天竟然才发现一直有一个误解 Proteced只能被同一个包内的或者子类的class访问 那么在另一个包的如下代码有问题吗? Sub sub = new Sub(); su ...

  7. tomcat启动非常慢

    解决: 有两种解决办法: 1)在Tomcat环境中解决 可以通过配置JRE使用非阻塞的Entropy Source. 在catalina.sh中加入这么一行: JAVA_OPTS="-Dja ...

  8. Electron、Node.js、JavaScript、JQuery、Vue.js、Angular.js,layui,bootstrap

    转载:https://blog.csdn.net/meplusplus/article/details/79033786 layui :是基于jquery库的封装开发. bootstrap:同样基于 ...

  9. shell 获取hive表结构

    hive -S -e "select * from db_name.table_name limit 0"|grep table_name|xargs -n1|sed 's/tab ...

  10. 15-----BBS论坛

    BBS论坛(十五) 15.1.登录界面完成 (1)front/signbase.html {% from 'common/_macros.html' import static %} <!DOC ...