好题。。写了两个半小时hh,省选的时候要一个半小时内调出这种题目还真是难= =

题目大意是给一棵树或环套树,求点距大于等于K的点对数

这里的树状数组做了一点变换。不是向上更新和向下求和,而是反过来,所以求和的时候sum(k)实际上是求k到n的和

所以我们要求大于等于k的dis的次数和,就是求sum(1,k-1),注意k要减一

如果是树,就是常规的点分治,然后用树状数组维护dis【t】出现的次数

如果是环套树,找环之后割掉一条边,然后先求这棵树的答案。接着考虑过了这条割掉的边s--t的情况:我们以这条边的一点t为起点,对于环上的每个点(即每棵子树的根),我们求出这棵子树的所有dis后,dis+cir_len-i为所求链的第一部分,链的第二部分的长度为k-(dis+cir_len-i),用树状数组求就可以了。更新树状数组的时候不是更新dis,而是dis+i;i即根到割的那条边的另一个点s的距离&&这条割边

完美解决。。然而常数还是很大,跑了两秒多

 #include<stdio.h>
 #include<string.h>
 #include<algorithm>
 #define INF 0x3f3f3f3f
 #define LL long long
 using namespace std;
 ;
 struct node{
     int to,next;
 }e[maxn*];
 int n,m,K,head[maxn],size[maxn],vis[maxn],sz,total,root,dis[maxn],tot,fa[maxn];
 ;
 LL p[maxn*],ans;

 void insert(int u, int v){
     e[++tot].to=v; e[tot].next=head[u]; head[u]=tot;
 }

 void add(int x, LL c){
     for (;x;x-=x&-x) p[x]+=c;
 }
 LL query(int x){  //注意:这里的树状数组是倒过来的, query(1,k) 是求得k+1到n
     LL ret=;
     ) x=;
     *n;x+=x&-x) ret+=p[x];
     return ret;
 }

 void getroot(int u, int f){
     size[u]=; ;
     for (int v,i=head[u]; i; i=e[i].next){
         if (vis[v=e[i].to] || v==f || i==ban1 || i==ban2) continue;
         getroot(v,u);
         size[u]+=size[v];
         mx=max(mx,size[v]);
     }
     mx=max(mx,total-size[u]);
     if (mx<sz) sz=mx,root=u;
 }

 void getdis(int u, int f, int d){
     dis[++tot]=d;
     for (int i=head[u],v; i; i=e[i].next){
         if (vis[v=e[i].to] || v==f || i==ban1 || i==ban2) continue;
         getdis(v,u,d+);
     }
 }

 void work(int u){
     total=size[u]?size[u]:n;
     sz=INF;
     getroot(u,); u=root;
     vis[u]=; tot=;
     ; i; i=e[i].next){
         if (vis[v=e[i].to] || i==ban1 || i==ban2) continue;
         last=tot;
         getdis(v,,); //printf("%d\n", tot);
         ; j<=tot; j++) ans+=query(K--dis[j]);
         ; j<=tot; j++) add(dis[j],);
     }
     ans+=query(K-);
     );
     for (int v,i=head[u]; i; i=e[i].next)
         if (!vis[v=e[i].to] && i!=ban1 && i!=ban2) work(v);
 }

 void find_cir(int u, int f){
     vis[u]=; if (len) return;//printf("  %d\n", u);
     for (int i=head[u],v; i; i=e[i].next){
         v=e[i].to;
         if (v==f || len) continue;
         fa[v]=u;// printf("now %d\n", u);
         if (vis[v]){
             ban1=i; ban2=i^;
             for (int x=fa[v]; x!=v; x=fa[x]) cir[++len]=x; cir[++len]=v;
             return;
         }
         find_cir(v,u);
     }
 }

 void cut(){
     ; i<=n; i++) vis[i]=;
     work();// printf("  %lld\n", ans);
     ; i<=n; i++) p[i]=0LL,vis[i]=;
     ; i<=len; i++) vis[cir[i]]=;
     ; i<=len; i++){
         ;
         getdis(u,,); //printf("  %d\n", tot);
         ; j<=tot; j++) ans+=query(K-dis[j]-(len-i+));//, printf("%lld\n", ans);
         );
     }
 }

 int main(){
     scanf(;
     ,u,v; i<=m; i++){
         scanf("%d%d", &u, &v);
         insert(u,v); insert(v,u);
     }
     ) work();
     else{
         find_cir(,);
         cut();
     }
     printf("%lld\n", ans);
     ;
 } 

bzoj3648: 寝室管理(环套树+点分治)的更多相关文章

  1. 【BZOJ-3648】寝室管理 环套树 + 树状数组 + 点分治

    3648: 寝室管理 Time Limit: 40 Sec  Memory Limit: 512 MBSubmit: 239  Solved: 106[Submit][Status][Discuss] ...

  2. BZOJ3648 寝室管理 【点分治 + 环套树】

    3648: 寝室管理 Time Limit: 40 Sec  Memory Limit: 512 MB Submit: 366  Solved: 152 [Submit][Status][Discus ...

  3. BZOJ3648 : 寝室管理

    求环套外向树上节点数不小于K的路径数. 首先树的话直接点分治+树状数组$O(n\log^2n)$搞定 环套树的话,先删掉多余的边(a,b) 然后变成了一棵树,直接点分治 然后在树上找到a到b的路径,将 ...

  4. 【bzoj3648】环套树+点分治+树状数组

    tree 1s 128M  by hzw czy神犇种了一棵树,他想知道地球的质量 给定一棵n个点的树,求树上经过点的个数≥K的路径数量ans 对于部分数据,树上某两点间会多出最多一条无向边 输入数据 ...

  5. 【BZOJ3648】寝室管理 树分治

    [BZOJ3648]寝室管理 Description T64有一个好朋友,叫T128.T128是寄宿生,并且最近被老师叫过去当宿管了.宿管可不是一件很好做的工作,碰巧T128有一个工作上的问题想请T6 ...

  6. BZOJ 3648: 寝室管理( 点分治 + 树状数组 )

    1棵树的话, 点分治+你喜欢的数据结构(树状数组/线段树/平衡树)就可以秒掉, O(N log^2 N). 假如是环套树, 先去掉环上1条边, 然后O(N log^2 N)处理树(同上); 然后再O( ...

  7. [ARC083F] Collecting Balls [建二分图+环套树定向+建拓扑图+树的拓扑序计数]

    题面 [传送门](https://arc083.contest.atcoder.jp/tasks/arc083_d) 思路 这是一道真正的好题 第一步:转化模型 行列支配类的问题,常见做法就是把行和列 ...

  8. 【BZOJ-1040】骑士 树形DP + 环套树 + DFS

    1040: [ZJOI2008]骑士 Time Limit: 10 Sec  Memory Limit: 162 MBSubmit: 3312  Solved: 1269[Submit][Status ...

  9. 【BZOJ】1040: [ZJOI2008]骑士(环套树dp)

    http://www.lydsy.com/JudgeOnline/problem.php?id=1040 简直不能再神的题orz. 蒟蒻即使蒟蒻,完全不会. 一开始看到数据n<=1000000就 ...

随机推荐

  1. luac++

    Cocos2d-x下Lua调用自定义C++类和函数的最佳实践 洪亮 305 2014年08月09日 发布   1 推荐 21 收藏,14.9k 浏览 关于cocos2d-x下Lua调用C++的文档看了 ...

  2. 修改Sqlserver实例默认排序规则

    1.将sqlserver安装盘加载到虚拟光驱,这里加载到F:盘跟目录 2.cmd进入命令 3.输入命令: F:/Setup /QUIET /ACTION=REBUILDDATABASE /INSTAN ...

  3. Linux Memcached安装以及PHP扩展安装

    一:安装libevent 由于memcached安装时,需要使用libevent类库,所以先安装libevent 1.下载 #wget   http://www.monkey.org/~provos/ ...

  4. Linux学习日记之磁盘与档案系统

    主要定义 磁盘的物理组成磁盘主要由圆形磁盘(多张).机械手臂.磁头等组成.每张磁盘都有不同的磁道,半径相同的磁道组成了磁柱,沿着中心划线可将磁盘分成若干扇区,每个扇区的大小是512Bytes. 磁盘分 ...

  5. php单点登录之模拟淘宝天猫同步登录

    说到单点登录大家都很了解,一个站点登录其他域会自动登录. 单点登录SSO(Single Sign On)的方法有很多,比如:p3p.共享session.共享cookice.第三方OAuth认证. 这里 ...

  6. SpringMVC(二) SpringMVC Hello World

    准备条件: STS(集成了Spring相关工具的Eclipse) Spring软件包 spring-framework-4.3.3.RELEASE-dist.zip. 步骤: 加入jar包. Ecli ...

  7. nodejs复习04

    TCP/UDP网络应用 创建TCP服务器客户端 socket套接字对象实例,是对TCP协议的一个基本封装接口 clientt套接字对象实例 //server.js var net = require( ...

  8. 商贸食品车销成功应用PDA抄单 现场开单 打印销售单安卓智能手持POS应用

    中小超市配送食品,酒水饮料,业务员以往是挨家挨户抄每个超市需要哪些东西,晚上回公司再统计,打到软件里面,开单配货. 选用PDA后,人手一台,直接在超市里面抄好货物,通过网络传输到公司软件上面,加快了工 ...

  9. Delphi 中的自动释放策略-转

    八.使用结构体而不是结构体指针: 很重要 一.指定 Owner 后, 随 Owner 连带释放: //uses Vcl.StdCtrls, Vcl.ExtCtrls; var panel: TPane ...

  10. Cucumber(一): Preparation

    Every time I wrote some code in ruby and executed our cucumber features I craved for something simil ...