今天算是把LCA这个坑填上了一点点,又复习(其实是预习)了一下树上差分。其实普通的差分我还是会的,树上的嘛,也是懂原理的就是没怎么打过。

我们先来把树上差分能做到的看一下:

1.找所有路径公共覆盖的边

例题:[NOIP2015]运输计划 (然而我还没过就先不讲了)

反正就是中间有一步要求一条边被所有计划公共覆盖。

那么怎么求它呢?暴力(滚粗)。我们有一个非常好的方法就是树上差分(记录tmp为差分数组)

询问操作为从叶子节点的权值向上累加到root

在一条路径u→ v,如果tmp[u]++,那么我们往上推的时候相当于u到root所有路径都被访问一次。同理tmp[v]++也意味如此。但是,lca(u,v)到root的路径都没有被访问过,但这里都被标记过两次,所以我们还要做的操作就是tmp[lca(u,v)]-=2;这样的话累加完之后tmp[i]记录的就是i节点被多少条路径覆盖了。

2.将路径上的所有点权值+1,最后求点权

例题:[JLOI2014]松鼠的新家 (这个我做过了hhh)

题目大意就是给你一些路径,把这个路径经过的点权+1,最后求所有点权。

这个题今天卡了我了。同学大佬有拿树剖求的,而且还要差分。但是我对于树上差分有点蒙蔽,于是搜了搜。然而蒟蒻的我搜到了LCA解法,于是兴高采烈的打(chao)了(le)出来。这里的差分有一些不同。因为我们要找的是点的覆盖。所以我们对于u→ v,tmp[u]++,tmp[v]++,tmp[lca(u,v)]--,tmp[fa[lca(u,v)]]--;这个想必大家能看懂吧。

于是,我们就欢快的求出了所有点被修改后的权值。

  1. #include<iostream>
  2. #include<algorithm>
  3. #include<cstdio>
  4. #include<cstring>
  5. #include<queue>
  6. #include<cmath>
  7. #define pos(i,a,b) for(int i=(a);i<=(b);i++)
  8. using namespace std;
  9. #define N 301000
  10. struct haha{
  11. int next,to;
  12. }edge[N*2];
  13. int head[N],cnt=1,p[N][20];
  14. void add(int u,int v){
  15. edge[cnt].to=v;edge[cnt].next=head[u];head[u]=cnt++;
  16. }
  17. int a[N],n,fa[N],dep[N];
  18. void dfs(int x){
  19. for(int i=head[x];i;i=edge[i].next){
  20. int to=edge[i].to;
  21. if(fa[x]!=to){
  22. fa[to]=x;dep[to]=dep[x]+1;
  23. dfs(to);
  24. }
  25. }
  26. }
  27. void init(){
  28. int j;
  29. for(j=0;(1<<j)<=n;j++){
  30. pos(i,1,n) p[i][j]=-1;
  31. }
  32. pos(i,1,n){
  33. p[i][0]=fa[i];
  34. }
  35. for(j=1;(1<<j)<=n;j++){
  36. pos(i,1,n){
  37. if(p[i][j-1]!=-1){
  38. p[i][j]=p[p[i][j-1]][j-1];
  39. }
  40. }
  41. }
  42. }
  43. int lca(int a,int b){
  44. int i;
  45. if(dep[a]<dep[b]) swap(a,b);
  46. for(i=0;(1<<i)<=dep[a];i++);
  47. i--;
  48. for(int j=i;j>=0;j--)
  49. if(dep[a]-(1<<j)>=dep[b])
  50. a=p[a][j];
  51. if(a==b) return a;
  52. for(int j=i;j>=0;j--){
  53. if(p[a][j]!=-1&&p[a][j]!=p[b][j]){
  54. a=p[a][j];b=p[b][j];
  55. }
  56. }
  57. return fa[a];
  58. }
  59. int tmp[N];
  60. void work(int x){
  61. for(int i=head[x];i;i=edge[i].next){
  62. int to=edge[i].to;
  63. if(fa[x]!=to){
  64. work(to);
  65. tmp[x]+=tmp[to];
  66. }
  67. }
  68. }
  69. int main(){
  70. scanf("%d",&n);
  71. pos(i,1,n)
  72. scanf("%d",&a[i]);
  73. pos(i,1,n-1){
  74. int x,y;scanf("%d%d",&x,&y);
  75. add(x,y);add(y,x);
  76. }
  77. dfs(a[1]);
  78. init();
  79. pos(i,1,n-1){
  80. int u=a[i],v=a[i+1];
  81. tmp[u]++;tmp[v]++;
  82. tmp[lca(u,v)]--;tmp[fa[lca(u,v)]]--;
  83. }
  84. work(a[1]);
  85. pos(i,2,n) tmp[a[i]]--;
  86. pos(i,1,n) printf("%d\n",tmp[i]);
  87. return 0;
  88. }

  

[填坑]树上差分 例题:[JLOI2014]松鼠的新家(LCA)的更多相关文章

  1. [JLOI2014] 松鼠的新家 (lca/树上差分)

    [JLOI2014]松鼠的新家 题目描述 松鼠的新家是一棵树,前几天刚刚装修了新家,新家有n个房间,并且有n-1根树枝连接,每个房间都可以相互到达,且俩个房间之间的路线都是唯一的.天哪,他居然真的住在 ...

  2. P3258[JLOI2014]松鼠的新家(LCA 树上差分)

    P3258 [JLOI2014]松鼠的新家 题目描述 松鼠的新家是一棵树,前几天刚刚装修了新家,新家有n个房间,并且有n-1根树枝连接,每个房间都可以相互到达,且俩个房间之间的路线都是唯一的.天哪,他 ...

  3. 【bzoj3631】[JLOI2014]松鼠的新家 LCA+差分数组

    题目描述 松鼠的新家是一棵树,前几天刚刚装修了新家,新家有n个房间,并且有n-1根树枝连接,每个房间都可以相互到达,且俩个房间之间的路线都是唯一的.天哪,他居然真的住在“树”上.松鼠想邀请小熊维尼前来 ...

  4. 【洛谷】【lca+树上差分】P3258 [JLOI2014]松鼠的新家

    [题目描述:] 松鼠的新家是一棵树,前几天刚刚装修了新家,新家有n(2 ≤ n ≤ 300000)个房间,并且有n-1根树枝连接,每个房间都可以相互到达,且俩个房间之间的路线都是唯一的.天哪,他居然真 ...

  5. luoguP3258 [JLOI2014]松鼠的新家 题解(树上差分)

    P3258 [JLOI2014]松鼠的新家  题目 树上差分:树上差分总结 #include<iostream> #include<cstdlib> #include<c ...

  6. [Bzoj3631][JLOI2014]松鼠的新家 (树上前缀和)

    3631: [JLOI2014]松鼠的新家 Time Limit: 10 Sec  Memory Limit: 128 MBSubmit: 2350  Solved: 1212[Submit][Sta ...

  7. 洛谷 P3258 [JLOI2014]松鼠的新家 树链剖分+差分前缀和优化

    目录 题面 题目链接 题目描述 输入输出格式 输入格式 输出格式 输入输出样例 输入样例: 输出样例: 说明 说明 思路 AC代码 优化 优化后AC代码 总结 题面 题目链接 P3258 [JLOI2 ...

  8. P3258 [JLOI2014]松鼠的新家

    P3258 [JLOI2014]松鼠的新家倍增lca+树上差分,从叶子节点向根节点求前缀和,dfs求子树和即可,最后,把每次的起点和终点都. #include<iostream> #inc ...

  9. [Luogu 3258] JLOI2014 松鼠的新家

    [Luogu 3258] JLOI2014 松鼠的新家 LCA + 树上差分. 我呢,因为是树剖求的 LCA,预处理了 DFN(DFS 序),于是简化成了序列差分. qwq不讲了不讲了,贴代码. #i ...

随机推荐

  1. 从 JavaScript 到 TypeScript

    本文首发在我的个人博客:http://muyunyun.cn/posts/66a54fc2/ 文中的案例代码已经上传到 TypeScript TypeScript 并不是一个完全新的语言, 它是 Ja ...

  2. .NET C#到Java没那么难,MVC篇

    最典型的JAVA MVC就是JSP + servlet + javabean的模式.比较好的MVC,老牌的有Struts.Webwork.新兴的MVC 框架有Spring MVC.Tapestry.J ...

  3. powerdesinger(MSSQLSRV2008测试通过)通过Name或comment 导出注释到sql脚本,生成sql的说明备注,包括表注释信息

    导出字段信息name注释到sql2008字段的说明 在database -> edit current dbms -> MSSQLSRV2008::Script\Objects\Colum ...

  4. servlet导出Excel

    package khservlet; import java.io.IOException;import java.io.PrintWriter;import java.sql.*; import j ...

  5. 分享一次Oracle数据导入导出经历

    最近工作上有一个任务要修改一个比较老的项目,分公司这边没有这个项目数据库相关的备份,所以需要从正式环境上面导出数据库备份出来在本地进行部署安装,之前在其它项目的时候也弄过这个数据库的部署和安装,也写了 ...

  6. 使用ajax提交form表单(转)

    前言 使用ajax请求数据,很多人都会,比如说: $.post(path,{data:data},function(data){ ... },"json"); 又或者是这样的aja ...

  7. .NetCore~C#6的一些新特性

    回到目录 在进行.netCore平台后,由于它的版本在.net4.6,C#6之后,所以它的语法也有一些新的特性,主要表现在以下几个方面 只读属性初始化 static string Hello => ...

  8. Linux系统下基本命令

    <Linux系统下基本命令> Linux系统下基本命令: 要区分大小写 uname 显示版本信息(同win2K的 ver) dir 显示当前目录文件,ls -al 显示包括隐藏文件(同wi ...

  9. 类 java.util.Scannar方法

    类 java.util.Scannar方法 ·Scannar (InputStream ln):用给定的输人流创建一个Scanner对象. ·String nextLlne():读取输入的下一行内容. ...

  10. 第一篇:webservice初探

    接触webservice也有一段时间了,为了查缺补漏,把知识点系统化,准备写几篇博文梳理下webservice的知识点,这是第一篇,对webservice进行大致的介绍. 1.什么是webservic ...