HDU 5293
树上DP题。
其实有点类似于01的问题。方程很容易想到。首先,因为一条链的节点其实都是在树上的,所以很容易想到应该先求一个LCA。然后,当某节点不是链的LCA时,它的转移就是:
dp[i]=sum[i],其中,sum[i]是i的子节点的dp[i]的和。
如果它是某个点的LCA时,那么它的转移就是dp[i]=val[p]+(sum(sum[k])-sum(dp[k except i])),k是一条chain上的节点。
然而,这样做是不会过的,TLE。。。
树状数组维和chain上的节点的和。QAQ.....
怎么样维护呢?首先,使用DFS把树处理成DFS序,并维护时间戳,这是很显然的,记为l[u],r[u]。然后,假设对于某条链的LCA为u,计算出sum[u]和dp[u]后,对l[u]~r[u]进行区间更新。为什么要区间更新呢?因为,如果以后存在一条chain的起点是在u的子树内,那么,我们在计算链上节点和时,该链必定是经过u点的,这样,求前缀和就可以得到一个和是包含了u点的。区间更新,单点查询。单点查询就是完成了从起或终点到它们的LCA的链上节点求和,因为每次区间更新的时候,因为起点都在子树内,就可以对链上的节点的和不停地累加。
我弱菜没想到优化,TLE的代码。。
#include <iostream>
#include <cstdio>
#include <cstring>
#include <algorithm>
#include <vector>
#pragma comment(linker, "/STACK:1024000000,1024000000")
using namespace std; const int MAX=100010; struct Point{
int v,i;
Point(int vv,int ii){
v=vv,i=ii;
}
}; vector<Point>chain[MAX];
vector<int>head_chain[MAX];
//vector<int>chain_point;
int chain_from[MAX],chain_to[MAX],chain_val[MAX];
int pre[MAX]; struct Edge{
int u,v,next;
}edge[MAX*2];
int head[MAX],tot,n,m;
int sum[MAX],dp[MAX];
int parent[MAX],depth[MAX];
bool color[MAX]; void addedge(int u,int v){
edge[tot].u=u;
edge[tot].v=v;
edge[tot].next=head[u];
head[u]=tot++;
} int get_point(int i){
// chain_point.clear();
int res=0;
int u=chain_from[i],v=chain_to[i];
if(depth[u]<depth[v]) swap(u,v);
while(depth[u]>depth[v]){
// chain_point.push_back(u);
res+=(sum[u]-dp[u]);
u=parent[u];
}
while(u!=v){
// chain_point.push_back(u);
// chain_point.push_back(v);
res+=(sum[u]-dp[u]);
res+=(sum[v]-dp[v]);
u=parent[u];
v=parent[v];
}
res+=sum[u];
// chain_point.push_back(u);
return res;
} void dfs(int u,int par){
sum[u]=0;
for(int e=head[u];e!=-1;e=edge[e].next){
int v=edge[e].v;
if(v!=par){
dfs(v,u);
sum[u]+=dp[v];
}
}
dp[u]=sum[u];
if(head_chain[u].size()){
int sz=head_chain[u].size();
int tmp,index,tsz;
for(int i=0;i<sz;i++){
index=head_chain[u][i];
tmp=get_point(index);
// tmp=0; tsz=chain_point.size();
// for(int k=0;k<tsz;k++){
// tmp+=(sum[chain_point[k]]-dp[chain_point[k]]);
// }
dp[u]=max(dp[u],tmp+chain_val[index]);
}
}
} int findx(int u){
int x=u;
while(pre[u]!=-1){
u=pre[u];
}
while(pre[x]!=-1){
int t=pre[x];
pre[x]=u;
x=t;
}
return u;
} void DFS(int u,int par,int dep){
parent[u]=par;
depth[u]=dep;
for(int e=head[u];e!=-1;e=edge[e].next){
int v=edge[e].v;
if(v!=par){
DFS(v,u,dep+1);
pre[findx(v)]=u;
}
}
color[u]=true;
int sz=chain[u].size(),v;
for(int i=0;i<sz;i++){
v=chain[u][i].v;
if(color[v]){
head_chain[findx(v)].push_back(chain[u][i].i);
}
}
} /*
void get_head(int i){
int u=chain_from[i],v=chain_to[i];
if(depth[u]<depth[v]) swap(u,v);
while(depth[u]>depth[v]){
u=parent[u];
}
while(u!=v){
u=parent[u],v=parent[v];
}
head_chain[u].push_back(i);
}
*/
int main(){
int T,u,v;
scanf("%d",&T);
while(T--){
scanf("%d%d",&n,&m);
tot=0;
memset(head,-1,sizeof(head));
memset(pre,-1,sizeof(pre));
memset(color,false,sizeof(color));
for(int i=1;i<n;i++){
scanf("%d%d",&u,&v);
addedge(u,v);
addedge(v,u);
head_chain[i].clear();
chain[i].clear();
}
chain[n].clear();
head_chain[n].clear();
for(int i=1;i<=m;i++){
scanf("%d%d%d",&chain_from[i],&chain_to[i],&chain_val[i]);
chain[chain_from[i]].push_back(Point(chain_to[i],i));
chain[chain_to[i]].push_back(Point(chain_from[i],i));
// get_head(i);
}
DFS(1,1,1);
dfs(1,0);
printf("%d\n",dp[1]);
}
return 0;
}
http://blog.csdn.net/qq_24451605/article/details/47003497
简单的代码,其实写起来真的不难,主要是优化没想到。。。。
#pragma comment(linker, "/STACK:1024000000,1024000000")
#include <iostream>
#include <cstdio>
#include <algorithm>
#include <cstring>
#include <vector>
#define MAX 200007 using namespace std; int n,m,t; typedef long long LL; LL d[MAX];
LL sum[MAX]; LL c1[MAX<<1];
LL c2[MAX<<1]; int lowbit ( int x )
{
return x&-x;
} void add1 ( int x , LL v )
{
while ( x <= n )
{
c1[x] += v;
x += lowbit ( x );
}
} void add2 ( int x , LL v )
{
while ( x <= n )
{
c2[x] += v;
x += lowbit ( x );
}
} LL sum1 ( int x )
{
LL res = 0;
while ( x )
{
res += c1[x];
x -= lowbit ( x );
}
return res;
} LL sum2 ( int x )
{
LL res = 0;
while ( x )
{
res += c2[x];
x -= lowbit ( x );
}
return res;
} typedef pair<int,int> PII; vector<int> e[MAX];
vector<int> chain[MAX];
vector<PII> a[MAX];
vector<LL> w[MAX];
vector<LL> val[MAX]; int fa[MAX];
int times;
bool used[MAX];
int l[MAX];
int r[MAX]; int _find ( int x )
{
return fa[x] == x ? x: fa[x] = _find ( fa[x]);
} void LCA ( int u )
{
fa[u] = u;
l[u] = ++times;
used[u] = true;
for ( int i = 0 ; i < e[u].size() ; i++ )
{
int v = e[u][i];
if ( used[v] ) continue;
LCA ( v );
fa[v] = u;
}
for ( int i = 0 ; i < chain[u].size() ; i++ )
{
int v = chain[u][i];
if ( !used[v] ) continue;
int x = _find ( v );
a[x].push_back ( make_pair ( u , v ));
w[x].push_back ( val[u][i] );
}
r[u] = ++times;
} void dfs ( int u , int p )
{
sum[u] = 0;
d[u] = 0;
for ( int i = 0 ; i < e[u].size() ; i++ )
{
int v = e[u][i];
if ( v == p ) continue;
dfs ( v , u );
sum[u] += d[v];
}
for ( int i = 0 ; i < a[u].size() ; i++ )
{
int x = a[u][i].first;
int y = a[u][i].second;
LL temp = sum1(l[x]) + sum1(l[y]) + sum[u]
-sum2(l[x]) - sum2(l[y]);
d[u] = max ( temp + w[u][i] , d[u] );
}
d[u] = max ( d[u] , sum[u] );
add1 ( l[u] , sum[u] );
add1 ( r[u] , -sum[u] );
add2 ( l[u] , d[u] );
add2 ( r[u] , -d[u] );
} void init ( )
{
times = 0;
memset ( c1 , 0 , sizeof ( c1 ) );
memset ( c2 , 0 , sizeof ( c2 ));
memset ( used , 0 , sizeof ( used ));
for ( int i = 0 ; i < MAX ; i++ )
{
e[i].clear();
val[i].clear();
a[i].clear();
w[i].clear();
chain[i].clear();
}
} int main ( )
{
int u,v,x;
scanf ( "%d" , &t );
while ( t-- )
{
init();
scanf ( "%d%d" , &n , &m );
int nn = n-1;
while ( nn-- )
{
scanf ( "%d%d" , &u , &v );
e[u].push_back ( v );
e[v].push_back ( u );
}
n = n*2;
while ( m-- )
{
scanf ( "%d%d%d" , &u , &v , &x );
chain[u].push_back ( v );
chain[v].push_back ( u );
val[u].push_back ( x );
val[v].push_back ( x );
}
//cout <<"YES" << endl;
LCA ( 1 );
dfs ( 1, -1 );
printf ( "%I64d\n" , d[1] );
}
}
HDU 5293的更多相关文章
- [HDU 5293]Tree chain problem(树形dp+树链剖分)
[HDU 5293]Tree chain problem(树形dp+树链剖分) 题面 在一棵树中,给出若干条链和链的权值,求选取不相交的链使得权值和最大. 分析 考虑树形dp,dp[x]表示以x为子树 ...
- HDU 5293 Tree chain problem 树形dp+dfs序+树状数组+LCA
题目链接: http://acm.hdu.edu.cn/showproblem.php?pid=5293 题意: 给你一些链,每条链都有自己的价值,求不相交不重合的链能够组成的最大价值. 题解: 树形 ...
- HDU 5293 Annoying problem 树形dp dfs序 树状数组 lca
Annoying problem 题目连接: http://acm.hdu.edu.cn/showproblem.php?pid=5293 Description Coco has a tree, w ...
- (中等) HDU 5293 Tree chain problem,树链剖分+树形DP。
Problem Description Coco has a tree, whose vertices are conveniently labeled by 1,2,…,n.There are ...
- codeforces 671D Roads in Yusland & hdu 5293 Tree chain problem
dp dp优化 dfs序 线段树 算是一个套路.可以处理在树上取链的问题.
- HDU 5293 Tree chain problem
树状数组 + dp 设$f_i$表示以$i$为根的子树中的能选取的最大和,$sum_x$表示$\sum_{f_y}$ ($y$是$x$的一个儿子),这样子我们把所有给出的链按照两点的$lca$分组, ...
- HDU 5293 Tree chain problem 树形DP
题意: 给出一棵\(n\)个节点的树和\(m\)条链,每条链有一个权值. 从中选出若干条链,两两不相交,并且使得权值之和最大. 分析: 题解 #include <cstdio> #incl ...
- 刷题总结——Tree chain problem(HDU 5293 树形dp+dfs序+树状数组)
题目: Problem Description Coco has a tree, whose vertices are conveniently labeled by 1,2,…,n.There ar ...
- HDU 5293 Train chain Problem - 树链剖分(树状数组) + 线段树+ 树型dp
传送门 题目大意: 一颗n个点的树,给出m条链,第i条链的权值是\(w_i\),可以选择若干条不相交的链,求最大权值和. 题目分析: 树型dp: dp[u][0]表示不经过u节点,其子树的最优值,dp ...
随机推荐
- Vue组件库elementUI 在el-row 或 el-col 上使用@click无效失效,
问题: elementUI 在el-row 或者 el-col 上使用@click失效, 解决: 在click后面加上 .native .要使用@click.native=”handler()”才行, ...
- sessionStorage 的使用
sessionStorage 的使用: sessionStorage.removeItem("data"); sessionStorage.getItem("data&q ...
- Codeforces 792C
题意:给出一个由0到9数字构成的字符串,要求删去最少的数位,使得这个字符串代表的数能被3整除,同时要求不能有前导零,并且至少有一位(比如数字11,删去两个1后就没有数位了,所以不符合).如果能够处理出 ...
- [转] 64位Oracle 11g R2的客户端连接时报ORA-01019错误
本文转自:http://blog.csdn.net/downmoon/article/details/8038583 在Win8企业版64位环境下,连接Oracle11g 服务端,搞了整整两天,特将过 ...
- semantic、vue 使用分页组件和日历插件
最近正在试试semantic-ui,结合了vue,这里忍不住吐槽semantic和vue的友好度简直不忍直视,不过既然用了,这里就分享几个用到的插件了 1.分页组件(基于vue) var pageCo ...
- Linux文件系统inode、block解释权限(三)
利用文件系统的inode和block来分析文件(目录)的权限问题. 为什么读取一个文件还要看该文件路径所有目录的权限? 为什么目录的w权限具有删除文件的能力,而文件w权限不行. inode:记录文件的 ...
- Unity引擎 UGUI
Unity UGUI讲解 1.导入UI图片资源 2.设置参数: TextureType(纹理类型) 精灵 2D and UI SpriteMode(精灵模式) Single(单) multiple( ...
- Angular——tab切换案例
基本介绍 angular框架下的tab切换,相比较于之前的纯js写的代码,有一个很大的特点就是以数据为驱动,基本上不用搜索dom元素就可以实现效果 基本使用 (1)导航部分使用的是的状态使用的是ng- ...
- VMware 11安装Mac OS X 10.10 及安装Mac Vmware Tools.
先上一张效果图兴奋一下,博主穷屌丝一个,只能通过虚拟黑苹果体验下高富帅的生活,感觉超爽的,废话不多说的,直接上图了! 目录: 1.安装所需软件下载: 2.Mac OS X10.10 安装基本步骤: 3 ...
- 【LaTeX】对xelatex的中英文设置不同的字体
不建议用Ctex套装,不好用. 用MixTex+TexStudio! XeTeX处理中文非常方便,不需要任何设置,就能够使用系统中安装的TrueType和OpenType字体. MikTeX2.7中已 ...