题意:给定一棵n个节点的树,然后在给定m条边,去掉m条边中的一条和原树中的一条边,使得树至少分为两部分,问有多少种方案。

神题,一点也想不到做法,

首先要分析出加入一条边之后会形成环,形成环的话,如果去掉该边和环上面没有被其他环覆盖的边,那么便分为两部分了。

这样只需要记录每条边被环覆盖了几次即可,

用dp[u]表示u点的父边被覆盖了几次。

每次新加进来一条边(a,b) dp[a] ++ ,dp[b] ++ , dp[lca(a,b)] -= 2;

所有边处理完之后,遍历一边此树,同时转移状态 dp[u] += dp[v];

#define maxn 100005

struct node
{
int v,next;
};
node e[maxn * ];
int cnt ;
int head[maxn * ];
int num ;
int E[maxn * + ],L[maxn * + ],R[maxn * + ];
int f[maxn * + ][];
int dp[maxn];
void init()
{
cnt = ;
memset(head,-,sizeof(head));
}
void add(int u ,int v)
{
e[cnt].v = v ;
e[cnt].next = head[u];
head[u] = cnt ++ ;
e[cnt].v = u;
e[cnt].next = head[v] ;
head[v] = cnt ++ ;
return ;
}
void dfs(int u ,int fa,int dep)
{
E[++num] = u , L[num] = dep , R[u] = num;
for(int i = head[u] ;i != -; i = e[i].next)
if(e[i].v != fa)
{
int v = e[i].v;
dfs(v, u, dep + );
E[++num] = u , L[num] = dep;
}
return;
}
void initRMQ()
{
for(int i = ; i <= num ;i ++ ) f[i][] = i;
for(int j = ; (<<j) <= num ; j++ )
for(int i = ; i + (<<j) - <= num; i ++ )
if(L[f[i][j-]] < L[f[i+(<<(j-))][j-]]) //ps:注意下标
f[i][j] = f[i][j-];
else f[i][j] = f[i+(<<(j-))][j-];
}
int lca(int a ,int b)
{
a = R[a] , b = R[b];
if(a>b) swap(a,b);
int k = (log(1.0 + b - a )/log(2.0));
if(L[f[a][k]] < L[f[b-(<<k)+][k]] )
return E[f[a][k]];
else return E[f[b-(<<k)+][k]];
}
void dfs(int u,int fa)
{
for(int i = head[u] ; i != - ; i =e[i].next)
if(e[i].v != fa)
{
int v = e[i].v;
dfs(v,u);
dp[u] += dp[v];
}
return ;
}
int main()
{
int n,m;
int u,v;
while(scanf("%d%d",&n,&m)!=EOF)
{
init();
for(int i = ; i < n - ; i ++ )
{
scanf("%d%d",&u,&v);
add(u,v);
}
num = ;
memset(R,,sizeof(R));
dfs(,,);
initRMQ();
memset(dp,,sizeof(dp));
for(int i = ; i < m ; i ++ )
{
scanf("%d%d",&u,&v);
dp[u] ++ ;
dp[v] ++ ;
dp[lca(u,v)] -= ;
}
dfs(,);
int sum ;
sum = ;
for(int i = ;i <= n ; i++ ) // ps:1为根不要选。
{
if(dp[i] == )
sum ++;
else if(dp[i] == )
sum += m;
}
printf("%d\n",sum);
}
return ;
} /*
好题:
离线LCA + 树形DP 分析:加进来一条边(a,b),会形成一条环 a - lca(a,b) - b
如果去掉(a,b)和没有被其他环覆盖的边,那么就能把树分成两部分。
只要记录dp[u]表示u的负边被环覆盖了几次
对于新加进来的一条边(a,b) dp[a] ++ , dp[b] ++ , dp[lca(a,b)] -= 2;///
这样处理为完所有新加进来的边之后,在dfs一边,跑完状态转移:dp[u] += dp[v];
这里就可以把每条环上的边都依次加1,并且因为lca(a,b)的父边不再环上,且已经提前减去了2
这样转移正好可以把减掉的2消去。 注意:这里把lca离线处理成rmq,只需要O(n)的遍历和O(nlogn)的预处理即可。 ST表 并且这里以1为根,后面的状态转移也要以1为根,否则lca就白算了。 */

poj3417

poj3417 Network 树形Dp+LCA的更多相关文章

  1. SGU 149. Computer Network( 树形dp )

    题目大意:给N个点,求每个点的与其他点距离最大值 很经典的树形dp...很久前就想写来着...看了陈老师的code才会的...mx[x][0], mx[x][1]分别表示x点子树里最长的2个距离, d ...

  2. hdu_5293_Tree chain problem(DFS序+树形DP+LCA)

    题目连接:http://acm.hdu.edu.cn/showproblem.php?pid=5293 被这题打蹦了,看着题解写的,很是爆炸,确实想不到,我用的DFS序+LCA+树形DP,当然也可以写 ...

  3. Uva LA 3902 - Network 树形DP 难度: 0

    题目 https://icpcarchive.ecs.baylor.edu/index.php?option=com_onlinejudge&Itemid=8&page=show_pr ...

  4. poj 3417 树形dp+LCA

    思路:我以前一直喜欢用根号n分段的LCA.在这题上挂了,第一次发现这样的LCA被卡.果断改用Tarjan离线算法求LCA. 当前节点为u,其子节点为v.那么: 当以v根的子树中含有连接子树以外点的边数 ...

  5. poj3417 Network 树上差分+LCA

    题目传送门 题目大意:给出一棵树,再给出m条非树边,先割掉一条树边,再割掉一条非树边,问有几种割法,使图变成两部分. 思路:每一条 非树边会和一部分的树边形成一个环,分三种情况: 对于那些没有形成环的 ...

  6. POJ 3728 The merchant (树形DP+LCA)

    题目:https://vjudge.net/contest/323605#problem/E 题意:一棵n个点的树,然后有m个查询,每次查询找(u->v)路径上的两个数,a[i],a[j],(i ...

  7. Codeforces Round #343 (Div. 2) E. Famil Door and Roads (树形dp,lca)

    Famil Door's City map looks like a tree (undirected connected acyclic graph) so other people call it ...

  8. HDU4008 Parent and son(树形DP LCA)

    先记录以1为根时每个节点子树儿子节点的最大与次小值,询问x, y时,先判断x在不在y的子树范围内,若不在,结果为y的儿子结点,后继的最小值. 若x在y的子树范围内,若y儿子最小值是x的前驱,从次小值与 ...

  9. poj3417 LCA + 树形dp

    Network Time Limit: 2000MS   Memory Limit: 65536K Total Submissions: 4478   Accepted: 1292 Descripti ...

随机推荐

  1. APUE读书笔记-第13章-守护进程

    第13章 守护进程 13.1 引言 *守护进程也称精灵进程(daemon)是生存期较长的一种进程.它们常常在系统自举时启动,仅在系统关闭时才终止.因为它们没有控制终端,所以说它们是在后台运行的.UNI ...

  2. Git使用操作指南和GitHub

    本文记录Git的使用操作,把散落的记忆整理到一起.并介绍GitHub的使用. 使用Git代表着一种思想和境地,和SVN相比,不是技术上的差异有多么大,而是代表融入了一种新的生态环境.一种开放开源的心态 ...

  3. MySQL初步研究数据库

    我用的是环境Win7.开始学习PHP和MySQL,而买了这<Head First PHP & MySQL>,从能Head First Labs官网获得HeadFirst系列书籍的相 ...

  4. 第1周 SQL Server 如何执行一个查询

    原文:第1周 SQL Server 如何执行一个查询 大家好,欢迎来到第1周的SQL Server性能调优培训.在我们进入SQL Server性能调优里枯燥难懂的细节内容之前,我想通过讲解SQL Se ...

  5. OpenStack Dashboard

    OpenStackDashboard 为管理员和普通用户提供了一个图形化管理界面.用户能够通过该界面訪问.分配或者自己主动化分配基于云的资源.可扩展的设计使得与第三方产品和服务融合变得非常easy,比 ...

  6. 20140719中国互联网公司市值排名TOP20

    近期在找工作.关注了一下中国互联网公司的市值,实际情况跟想象的区别非常大. 比方异军突起的小米.京东.唯品会.聚美优品. 比方乐视网由于政策原因,市值两日缩水10亿$.停牌了. 搜房网市值90天蒸发3 ...

  7. cocos2d之Box2D详细说明 鼠标联合实现

    cocos2d之Box2D具体解释 鼠标关节实现 DionysosLai2014-5-7 我们常常要移动物理世界中的某个物体,例如说石头.木块等.假设我们直接改变这些物体的位置,让这些物体尾随我们手指 ...

  8. GUIForDebug

    package gui; import org.luaj.vm2.Globals; import org.luaj.vm2.LuaValue; import org.luaj.vm2.ast.Chun ...

  9. socket计划编制的原则

    socket编程原理 1.问题的引入 1) 普通的I/O操作过程: UNIX系统的I/O命令集,是从Maltics和早期系统中的命令演变出来的,其模式为打开一读/写一关闭(open-write-rea ...

  10. Reveal:分析iOS UI该武器

    Reveal是分析iOS应用UI的利器: Reveal可以在执行时调试和改动iOS应用程序.它能连接到应用程序,并同意开发人员编辑各种用户界面參数.这反过来会马上反应在程序的UI上.就像用FireBu ...