洛谷P3354 Riv河流 [IOI2005] 树型dp
正解:树型dp
解题报告:
简要题意:有棵树,每个节点有个权值w,要求选k个节点,最大化∑dis*w,其中如果某个节点到根的路径上选了别的节点,dis指的是到达那个节点的距离
首先这个一看就是个树型dp嘛,关键是怎么设状态
首先肯定是想到设f[i][j]:以i为根的子树中选了j个点
这个时候发现布星昂,这么设的时候我们不能得到对于到根的路径上的点的贡献鸭,所以就考虑加一维
所以f[i][j][k]:以i为根的子树中选了j个点,最近祖先的距离为k的最大代价
然后直接转移就好
对了还要说一个就,我这里有转点儿题意?就它是问最小化花费,然后我写题解的时候是转化成的tot-max∑
但实际实现的时候我是直着写的就是说直接最小化花费打的QAQ
差不多差不多QAQ
然后我发现我现在码力是真的弱,,,,我知道这题正解了,还是个dp题,然后打了2h?我原地自杀了要,,,
#include<bits/stdc++.h>
using namespace std;
#define il inline
#define fr first
#define sc second
#define rg register
#define gc getchar()
#define ll long long
#define mp make_pair
#define rp(i,x,y) for(rg int i=x;i<=y;++i)
#define my(i,x,y) for(rg int i=x;i>=y;--i) const int N=+,K=+;
int n,k,w[N],stck_top,stck[N],dis[N];
ll f[N][K][N][];
vector< pair<int,int> >son[N]; il int read()
{
rg char ch=gc;rg int x=;rg bool y=;
while(ch!='-' && (ch>'' || ch<''))ch=gc;
if(ch=='-')ch=gc,y=;
while(ch>='' && ch<='')x=(x<<)+(x<<)+(ch^''),ch=gc;
return y?x:-x;
}
il void dfs(int nw)
{
stck[++stck_top]=nw;
int sz=son[nw].size();
rp(i,,sz-)
{
dis[son[nw][i].fr]=dis[nw]+son[nw][i].sc;dfs(son[nw][i].fr);
my(j,stck_top,)
{
my(p,k,)
{
f[nw][p][stck[j]][]+=f[son[nw][i].fr][][stck[j]][];
f[nw][p][stck[j]][]+=f[son[nw][i].fr][][nw][];
my(q,p,)
{
f[nw][p][stck[j]][]=min(f[nw][p][stck[j]][],f[son[nw][i].fr][q][stck[j]][]+f[nw][p-q][stck[j]][]),
f[nw][p][stck[j]][]=min(f[nw][p][stck[j]][],f[son[nw][i].fr][q][nw][]+f[nw][p-q][stck[j]][]);
}
}
}
}
rp(i,,stck_top)
{
my(j,k,)
if(j)f[nw][j][stck[i]][]=min(f[nw][j-][stck[i]][],f[nw][j][stck[i]][]+w[nw]*(dis[nw]-dis[stck[i]]));
else f[nw][j][stck[i]][]+=w[nw]*(dis[nw]-dis[stck[i]]);
}
--stck_top;
return;
} int main()
{
// freopen("riv.in","r",stdin);freopen("riv.out","w",stdout);
n=read();k=read();
rp(i,,n){w[i]=read();int fa=read(),dis=read();son[fa].push_back(mp(i,dis));}
dfs();printf("%lld\n",f[][k][][]);
return ;
}
最后放下代码QAQ
顺便港下,这题有个双倍经验
都差不多只是还要建棵trie树就好了,over
等下把那题代码也放上来吼QwQ
#include<bits/stdc++.h>
using namespace std;
#define il inline
#define fr first
#define sc second
#define rg register
#define gc getchar()
#define ll long long
#define mp make_pair
#define rp(i,x,y) for(rg int i=x;i<=y;++i)
#define my(i,x,y) for(rg int i=x;i>=y;--i) const int N=+,K=+;
int n,k,stck_top,stck[N],dis[N],nod_cnt;
ll f[N][K][N][],as;
struct node{int to[],wei;}tr[N*]; il int read()
{
rg char ch=gc;rg int x=;rg bool y=;
while(ch!='-' && (ch>'' || ch<''))ch=gc;
if(ch=='-')ch=gc,y=;
while(ch>='' && ch<='')x=(x<<)+(x<<)+(ch^''),ch=gc;
return y?x:-x;
}
il void insert(string str,ll wei)
{
ll lth=str.length(),nw=;
rp(i,,lth-)
{
if(!tr[nw].to[str[i]^''])tr[nw].to[str[i]^'']=++nod_cnt;
nw=tr[nw].to[str[i]^''];tr[nw].wei+=wei;
}
}
il void dfs(int nw)
{
stck[++stck_top]=nw;
rp(i,,)
{
if(!tr[nw].to[i])continue;dis[tr[nw].to[i]]=dis[nw]+;dfs(tr[nw].to[i]);
my(j,stck_top,)
my(p,k,)
my(q,p,)
f[nw][p][stck[j]][]=max(f[nw][p][stck[j]][],f[tr[nw].to[i]][q][stck[j]][]+f[nw][p-q][stck[j]][]),
f[nw][p][stck[j]][]=max(f[nw][p][stck[j]][],f[tr[nw].to[i]][q][nw][]+f[nw][p-q][stck[j]][]);
}
rp(i,,stck_top)
my(j,k,)
if(j)f[nw][j][stck[i]][]=max(f[nw][j-][stck[i]][]+tr[nw].wei*(dis[nw]-dis[stck[i]]),f[nw][j][stck[i]][]);
--stck_top;
return;
} int main()
{
// freopen("sd.in","r",stdin);freopen("sd.out","w",stdout);
n=read();k=read();rp(i,,n){string str;cin>>str;ll wei=read();insert(str,wei);as+=str.length()*wei;}
dfs();printf("%lld\n",as-f[][k][][]);
return ;
}
因为一些,奇怪的问题,我发现我顺着做布星,按上面那个as-=max的思路就欧克,,,
洛谷P3354 Riv河流 [IOI2005] 树型dp的更多相关文章
- 洛谷 P1453 城市环路 ( 基环树树形dp )
题目链接 题目背景 一座城市,往往会被人们划分为几个区域,例如住宅区.商业区.工业区等等.B市就被分为了以下的两个区域--城市中心和城市郊区.在着这两个区域的中间是一条围绕B市的环路,环路之内便是B市 ...
- 洛谷 P3384 【模板】树链剖分-树链剖分(点权)(路径节点更新、路径求和、子树节点更新、子树求和)模板-备注结合一下以前写的题目,懒得写很详细的注释
P3384 [模板]树链剖分 题目描述 如题,已知一棵包含N个结点的树(连通且无环),每个节点上包含一个数值,需要支持以下操作: 操作1: 格式: 1 x y z 表示将树从x到y结点最短路径上所有节 ...
- 洛谷p3384【模板】树链剖分题解
洛谷p3384 [模板]树链剖分错误记录 首先感谢\(lfd\)在课上调了出来\(Orz\) \(1\).以后少写全局变量 \(2\).线段树递归的时候最好把左右区间一起传 \(3\).写\(dfs\ ...
- POJ3659 Cell Phone Network(树上最小支配集:树型DP)
题目求一棵树的最小支配数. 支配集,即把图的点分成两个集合,所有非支配集内的点都和支配集内的某一点相邻. 听说即使是二分图,最小支配集的求解也是还没多项式算法的.而树上求最小支配集树型DP就OK了. ...
- POJ 3342 - Party at Hali-Bula 树型DP+最优解唯一性判断
好久没写树型dp了...以前都是先找到叶子节点.用队列维护来做的...这次学着vector动态数组+DFS回朔的方法..感觉思路更加的清晰... 关于题目的第一问...能邀请到的最多人数..so ea ...
- 【XSY1905】【XSY2761】新访问计划 二分 树型DP
题目描述 给你一棵树,你要从\(1\)号点出发,经过这棵树的每条边至少一次,最后回到\(1\)号点,经过一条边要花费\(w_i\)的时间. 你还可以乘车,从一个点取另一个点,需要花费\(c\)的时间. ...
- 【POJ 3140】 Contestants Division(树型dp)
id=3140">[POJ 3140] Contestants Division(树型dp) Time Limit: 2000MS Memory Limit: 65536K Tot ...
- Codeforces 581F Zublicanes and Mumocrates(树型DP)
题目链接 Round 322 Problem F 题意 给定一棵树,保证叶子结点个数为$2$(也就是度数为$1$的结点),现在要把所有的点染色(黑或白) 要求一半叶子结点的颜色为白,一半叶子结点的 ...
- ZOJ 3949 (17th 浙大校赛 B题,树型DP)
题目链接 The 17th Zhejiang University Programming Contest Problem B 题意 给定一棵树,现在要加一条连接$1$(根结点)和$x$的边,求加 ...
随机推荐
- SmileyCount.java笑脸加法程序代写(QQ:928900200)
SmileyCount.java 1/4Java Programming 2014Course Code: EBU4201Mini ProjectTask 1 [30 marks]SmileyCoun ...
- Y460 安装ubuntu 12.04系统黑屏,登录界面黑屏
ubuntu 12.04系统黑屏,登录界面黑屏,但是命令行界面可以登录,也可以正常使用,当时在装CVS,装完重启就这样了,可能是因为前一天装更新时,突然断电导致图形界面损坏,参考他人方法,终于修复,总 ...
- 关于Unity中NGUI的帧动画和Tween动画
帧动画 1.把三张帧动画的贴图png制作成图集,NGUI---->Open---->Atlas Maker,生成一个预制体,一个材质球,一个大图 2.创建一个Sprite类型的Sprite ...
- GDC2017 把“现实的天空”在游戏内再现【Forza Horizon 3】的天空表现
原文链接 http://game.watch.impress.co.jp/docs/news/1047800.html 完全表现出现实世界中各种偶然而不可预料的风景! [Forza Horiz ...
- sqllocaldb 2016安装
msiexec /i SqlLocalDB.msi /qn IACCEPTSQLLOCALDBLICENSETERMS=YES
- 【转】在xcode5中修改整个项目名
本文转载自:http://www.cnblogs.com/tbfirstone/p/3601541.html 总会遇到几个项目,在做到一半的时候被要求改项目名,网上找了下相关的资料,大多数是xcode ...
- 用 SQLite 和 FMDB 替代 Core Data
本文转载至 http://blog.csdn.net/majiakun1/article/details/38680147 为什么我不使用Core Data Mike Ash 写到: 就个人而言,我不 ...
- js - 移动端的超出滚动功能,附带滚动条,可解决弹层中滚动穿透问题。
背景: 弹层里边有可滚动区域时,在移动端的坑我就不多说了. 找了很多解决滚动穿透的方案,最终都不能完美解决. 一气之下自己js撸了一个. 效果图: 原理: 1.解决滚动穿透:通过给弹层绑定touchm ...
- spark on yarn 无法提交任务问题
java.lang.NoClassDefFoundError: com/sun/jersey/api/client/config/ClientConfig spark任务提交出错. 原因: spark ...
- 01List.ashx(班级列表动态页面)
01List.html <!DOCTYPE html> <html xmlns="http://www.w3.org/1999/xhtml"> <he ...