洛谷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$的边,求加 ...
随机推荐
- Java、Linux、Win 快速生成指定大小的空文件
Linux dd 命令: dd if=/dev/zero of=<fileName> bs=<一次复制的大小> count=<复制的次数> 生成 50 MB 的空文 ...
- var_dump
1:var_dump 主要输出类型和.长度和变量的值: 2: var_dump() 能打印出类型print_r() 只能打出值echo() 是正常输出...需要精确调试的时候用 var_dump(); ...
- svn常见错误解决
Svn冲突导致锁住的解决方案:错误码:svn: E155037: Previous operation has not finished; run 'cleanup' if it was interr ...
- android( java) 处理 null 和 预防空指针异常(NullPointerException) 的一些经验。
概述: 在实际编码中总是会遇到 空指针异常 ,本文总结了一些处理空指针的个人经验. 原则: 尽早的检查,尽早的失败. 比如: 通过intent传参到新的目标 activity,而且一定需要这个参数,那 ...
- Netbeans rcp中获得本地文件系统路径
通过file协议 —————————————————————————————————————————————————————— URL url = new URL("file:///E:/A ...
- Oracle中Select语句完整的执行顺序
oracle Select语句完整的执行顺序: .from 子句组装来自不同数据源的数据: .where 子句基于指定的条件对记录行进行筛选: .group by子句将数据划分为多个分组: .使用聚集 ...
- centos下安装Loadrunner
背景: 网上的资料呀,真是浑水摸鱼的多,有些人直接拷贝别人的帖子,这样有啥意思呢,只会让别人要搜索的时候,更扰乱些! 这里我不写步骤,我用shell把步骤弄了一下,看的懂的看,看不懂的留言吧.就酱,看 ...
- windows server 2008 R2 计划任务备份系统
实验环境拓扑图: 实验效果: Windows Server Backup 可以设置备份计划,但只能按日进行备份,不能设置按周或月进行备份.所以,需要使用到 windows Server 2008 R2 ...
- myeclipse创建hibernate工程
1.创建数据库: from blog http://www.cnblogs.com/zhaocundang/p/9061959.html 使用navicat mysql IDE: 创建数据库 book ...
- HTTP Status 500 - Could not open Hibernate Session for transaction;
错误原因: mysql数据库没有连接上 我们来启动mysql 方法1: 管理员身份运行 cmd 输入: net start mysql 方法2: Windows + R 运行 services.ms ...