发展城市

Time Limit: 20 Sec  Memory Limit: 512 MB
[Submit][Status][Discuss]

Description

  众所周知,Hzwer学长是一名高富帅,他打算投入巨资发展一些小城市。
  Hzwer打算在城市中开N个宾馆,由于Hzwer非常壕,所以宾馆必须建在空中,但是这样就必须建立宾馆之间的连接通道。机智的Hzwer在宾馆中修建了N-1条隧道,也就是说,宾馆和隧道形成了一个树形结构。
  Hzwer有时候会花一天时间去视察某个城市,当来到一个城市之后,Hzwer会分析这些宾馆的顾客情况。对于每个顾客,Hzwer用三个数值描述他:(S, T, V)表示该顾客这天想要从宾馆S走到宾馆T,他的速度是V。
  Hzwer需要做一些收集一些数据,这样他就可以规划他接下来的投资。
  其中有一项数据就是收集所有顾客可能的碰面次数。
  每天清晨,顾客同时从S出发以V的速度前往T(注意S可能等于T),当到达了宾馆T的时候,顾客显然要找个房间住下,那么别的顾客再经过这里就不会碰面了。特别的,两个顾客同时到达一个宾馆是可以碰面的。同样,两个顾客同时从某宾馆出发也会碰面。

Input

  第一行一个正整数T(1<=T<=20),表示Hzwer发展了T个城市,并且在这T个城市分别视察一次。
   对于每个T,第一行有一个正整数N(1<=N<=10^5)表示Hzwer在这个城市开了N个宾馆。
   接下来N-1行,每行三个整数X,Y,Z表示宾馆X和宾馆Y之间有一条长度为Z的隧道
   再接下来一行M表示这天顾客的数量。
   紧跟着M行每行三个整数(S, T, V)表示该顾客会从宾馆S走到宾馆T,速度为v

Output

   对于每个T,输出一行,表示顾客的碰面次数。

Sample Input

  2
  3
  1 2 1
  2 3 1
  3
  1 3 2
  3 1 1
  1 2 3
  1
  0

Sample Output

  2
  0

HINT

  1<=T<=20   1<=N<=10^5   0<=M<=10^3   1<=V<=10^6   1<=Z<=10^3

Main idea

  给定若干个顾客,每个顾客会匀速从一个点走到另外一个点,问有几对人会在路上相遇。

Solution

  由于人数较少,所以我们可以O(m^2)判断两人是否相交。

  首先,两个人相遇只可能是在路径重叠的部分相遇,所以我们先对两人的路径求交,这里提供一种对树上路径求交的方法:

    求出AB两人路径的交:
      设A路径为a.u->a.v,B路径为b.u->b.v。那么我们求出 LCA(a.u,b.u),LCA(a.v,b.v),LCA(a.u,b.v),LCA(b.u,a.v),然后保留下在AB路径上的点。(判断一个点是否在路径上:若u在LCA(x,y)的子树中,且u为LCA(u,x)或者LCA(u,y),则u在路径x,y上),然后按照dfs序位置排序,去重,保留下后两个点,则后两个点即是路径的端点。

  但是这样求交的话需要多次查询LCA,我们用每次log的时间查询显然会超时,于是我们引进O(nlog(n))预处理,O(1)查询的LCA。

    O(1)查询的LCA:
      我们先求出对于这棵树的欧拉序(欧拉序:记下每个点第一次访问的位置以及回溯完的位置),然后我们用那么这时候x,y之间的LCA也就是 [pos[x],pos[y]] 区间内深度最小的点(pos[x]表示点x第一次在欧拉序中出现的位置),这个区间最小值用RMQ求即可。

  现在我们已经求出了路径的交集,然后我们暴力分类讨论一下。如果没有交集则必然不相交,若交于一点判断一下到达时间即可,否则:

  先考虑两人运动的方向。我们记录p.u,p.v表示路径交集的两个端点。A1表示A到先进入路径的端点,A2表示A后到的端点,B1、B2类似(用距离长短判断即可),如果A1=B1则表示两人同向运动,否则表示相向运动。然后我们讨论一下:

    同向运动:如果两人同向运动,那么若先进入路径的后离开路径,则两人会相遇。
    相向运动:如果两人相向运动,则我们记录到端点的时间,如果两个人在路径上的时间有交集的话,则会相遇。

  然后这样判断一下就可以求出答案了,但是由于double定义下的除法速度很慢,会被卡常数,所以我们再用在long long定义下的交叉相乘来判断以上情况。这样我们就解决了这道题\(≧▽≦)/

Code

 #include<iostream>
#include<algorithm>
#include<cstdio>
#include<cstring>
#include<cstdlib>
#include<cmath>
using namespace std;
typedef long long s64; const int ONE = ; int T;
int n,m;
int x,y,z;
int next[ONE*],first[ONE],go[ONE*],w[ONE*],tot;
int pos[ONE],dfn_cnt,Dep[ONE],size[ONE];
int MinD[ONE*][],NumD[ONE*][];
int Log[ONE*],Bin[];
int Stk[],top,fc;
int Ans;
s64 d[ONE]; struct power
{
int u,v;
int val;
}a[],p; namespace input
{
const int BufferSize = << | ; char buffer[BufferSize];
char *head = buffer + BufferSize;
const char *tail = head; inline char nextChar()
{
if (head == tail)
{
fread(buffer, , BufferSize, stdin);
head = buffer;
}
return *head++;
} inline int get()
{
static char c;
while ((c = nextChar()) < '' || c > ''); int res = c - '';
while ((c = nextChar()) >= '' && c <= '')
res = res * + c - '';
return res;
}
}
using input::get; inline void Add(int u,int v,int z)
{
next[++tot]=first[u]; first[u]=tot; go[tot]=v; w[tot]=z;
next[++tot]=first[v]; first[v]=tot; go[tot]=u; w[tot]=z;
} namespace F
{
int Dfs(int u,int father)
{
pos[u] = ++dfn_cnt;
Dep[u] = Dep[father] + ;
size[u] = ;
MinD[dfn_cnt][]=Dep[u]; NumD[dfn_cnt][]=u;
for(int e=first[u];e;e=next[e])
{
int v=go[e];
if(v==father) continue;
d[v] = d[u] + w[e];
Dep[v] = Dep[u] + ;
Dfs(v,u);
size[u] += size[v];
MinD[++dfn_cnt][]=Dep[u]; NumD[dfn_cnt][]=u;
}
} inline void Pre_Rmq()
{
for(int j=;j<=;j++)
for(int i=;i<=dfn_cnt;i++)
if(i+Bin[j]- <= dfn_cnt)
{
int Next = i + Bin[j-];
if(MinD[i][j-] < MinD[Next][j-])
MinD[i][j]=MinD[i][j-], NumD[i][j]=NumD[i][j-];
else
MinD[i][j]=MinD[Next][j-], NumD[i][j]=NumD[Next][j-];
}
else break;
}
} inline int LCA(int x,int y)
{
x=pos[x]; y=pos[y];
if(x > y) swap(x,y);
int T = Log[y - x +];
if(MinD[x][T] < MinD[y-Bin[T]+][T]) return NumD[x][T];
return NumD[y-Bin[T]+][T];
} inline s64 dist(int x,int y)
{
return d[x] + d[y] - (d[LCA(x,y)] << ) ;
} namespace PD
{
inline bool inroad(int u,power a)
{
int lca = LCA(a.u,a.v);
if(LCA(u,lca) != lca) return ;
return (LCA(u,a.u)==u || LCA(u,a.v)==u);
}
} inline void Sort(int n)
{
for(int i=;i<=n;i++)
for(int j=i+;j<=n;j++)
if(pos[Stk[i]] > pos[Stk[j]])
swap(Stk[i], Stk[j]);
} inline power Get_road(power a,power b)
{
fc=top=;
Stk[++fc] = LCA(a.u,b.u); Stk[++fc] = LCA(a.v,b.v);
Stk[++fc] = LCA(a.u,b.v); Stk[++fc] = LCA(b.u,a.v);
for(int i=;i<=fc;i++)
if(PD::inroad(Stk[i],a) && PD::inroad(Stk[i],b))
Stk[++top] = Stk[i]; Sort(top);
top=unique(Stk+,Stk+top+) - Stk - ; power p; p.val = ;
if(top==) p.u = p.v = ;
if(top==) p.u = p.v = Stk[];
if(top==) p.u=Stk[], p.v=Stk[];
if(top==) p.u=Stk[], p.v=Stk[];
return p;
} inline bool pmin(s64 a,s64 b,s64 c)
{
if(a<=b && b<=c) return ;
return ;
} inline int Deal(int x,int y)
{
if(a[x].u == a[y].u) return ;
p = Get_road(a[x],a[y]);
if(p.u==p.v)
{
if(!p.u) return ;
return (s64) dist(a[x].u,p.u) * a[y].val == (s64) dist(a[y].u,p.u) * a[x].val;
} int A1,A2,B1,B2;
double A1_time,A2_time,B1_time,B2_time;
if(dist(a[x].u,p.u) < dist(a[x].u,p.v)) A1=p.u, A2=p.v;else A1=p.v, A2=p.u;
if(dist(a[y].u,p.u) < dist(a[y].u,p.v)) B1=p.u, B2=p.v;else B1=p.v, B2=p.u; A1_time=(s64)dist(A1,a[x].u)*a[y].val; A2_time=(s64)dist(A2,a[x].u)*a[y].val;
B1_time=(s64)dist(B1,a[y].u)*a[x].val; B2_time=(s64)dist(B2,a[y].u)*a[x].val; if(A1==B1)//same
{
if(A1_time == B1_time) return ;
if(A1_time < B1_time) return A2_time >= B2_time;
return A2_time <= B2_time;
}
else
{
if(pmin(A1_time,B1_time,A2_time)) return ;
if(pmin(A1_time,B2_time,A2_time)) return ;
if(pmin(B1_time,A1_time,B2_time)) return ;
if(pmin(B1_time,A2_time,B2_time)) return ;
return ;
}
} inline void Solve()
{
n=get();
dfn_cnt=tot=;
memset(first,,sizeof(first));
for(int i=;i<n;i++)
{
x=get(); y=get(); z=get();
Add(x,y,z);
} F::Dfs(,); F::Pre_Rmq(); m=get();
for(int i=;i<=m;i++)
{
a[i].u=get(); a[i].v=get(); a[i].val=get();
} Ans = ; for(int i=;i<=m;i++)
for(int j=i+;j<=m;j++)
{
Ans += Deal(i,j);
} printf("%d\n",Ans);
} int main()
{
Log[]=-; for(int i=;i<=2e5;i++) Log[i] = Log[i>>] + ;
Bin[]=; for(int i=;i<=; i++) Bin[i] = Bin[i-] << ; T=get(); while(T--)
Solve();
}

【BZOJ3700】发展城市 [LCA][RMQ]的更多相关文章

  1. BZOJ3700: 发展城市

    BZOJ3700: 发展城市 https://lydsy.com/JudgeOnline/problem.php?id=3700 分析: 枚举两个人,先求链交,求到两个端点的时间. 链交求法:求两两\ ...

  2. BZOJ1906树上的蚂蚁&BZOJ3700发展城市——RMQ求LCA+树链的交

    题目描述 众所周知,Hzwer学长是一名高富帅,他打算投入巨资发展一些小城市. Hzwer打算在城市中开N个宾馆,由于Hzwer非常壕,所以宾馆必须建在空中,但是这样就必须建立宾馆之间的连接通道.机智 ...

  3. 发展城市 BZOJ 3700

    发展城市 [问题描述] 众所周知,Hzwer学长是一名高富帅,他打算投入巨资发展一些小城市. Hzwer打算在城市中开N个宾馆,由于Hzwer非常壕,所以宾馆必须建在空中,但是这样就必须建立宾馆之间的 ...

  4. 【Homework】LCA&RMQ

    我校是神校,作业竟然选自POJ,难道不知道“珍爱生命 勿刷POJ”么? 所有注明模板题的我都十分傲娇地没有打,于是只打了6道题(其实模板题以前应该打过一部分但懒得找)(不过感觉我模板还是不够溜要找个时 ...

  5. POJ 2763 (LCA +RMQ+树状数组 || 树链部分) 查询两点距离+修改边权

    题意: 知道了一颗有  n 个节点的树和树上每条边的权值,对应两种操作: 0 x        输出 当前节点到 x节点的最短距离,并移动到 x 节点位置 1 x val   把第 x 条边的权值改为 ...

  6. 算法详解(LCA&RMQ&tarjan)补坑啦!完结撒花(。◕ˇ∀ˇ◕)

    首先,众所周知,求LCA共有3种算法(树剖就不说了,太高级,以后再学..). 1.树上倍增(ST表优化) 2.RMQ&时间戳(ST表优化) 3.tarjan(离线算法)不讲..(后面补坑啦!) ...

  7. 牛客小白月赛13 小A的最短路(lca+RMQ)

    链接:https://ac.nowcoder.com/acm/contest/549/F来源:牛客网 题目描述 小A这次来到一个景区去旅游,景区里面有N个景点,景点之间有N-1条路径.小A从当前的一个 ...

  8. UESTC 912 树上的距离 --LCA+RMQ+树状数组

    1.易知,树上两点的距离dis[u][v] = D[u]+D[v]-2*D[lca(u,v)] (D为节点到根节点的距离) 2.某条边<u,v>权值一旦改变,将会影响所有以v为根的子树上的 ...

  9. [LCA & RMQ] [NOIP2013] 货车运输

    首先看到这题, 由于要最大, 肯定是求最大生成树 那么 o(n2) dfs 求任意点对之间的最小边是可以想到的 但是看看数据范围肯定TLE 于是暴力出来咯, 不过要注意query的时候判断的时候要 m ...

随机推荐

  1. Yarn 调度器Scheduler详解

    理想情况下,我们应用对Yarn资源的请求应该立刻得到满足,但现实情况资源往往是有限的,特别是在一个很繁忙的集群,一个应用资源的请求经常需要等待一段时间才能的到相应的资源.在Yarn中,负责给应用分配资 ...

  2. OBS源码编译开发

    本文来自网易云社区 作者:梁敏 OBS简介 OBS(Open Broadcaster Software)是免费开源的视频录制和直播软件,支持运行在windows,Mac和linux平台.官方链接 ht ...

  3. Android中通过拨号调起应用的实现方式及特殊情况处理

    Android中有时我们会有这样的需求:通过拨号调起我们的程序.这个需求如何实现呢? 思路当然是在我们的应用中实现一个广播接收器(BroadcastReceiver),对打电话时系统发出的广播进行拦截 ...

  4. 【题解搬运】PAT_A1016 Phone Bills

    从我原来的博客上搬运.原先blog作废. 题目 A long-distance telephone company charges its customers by the following rul ...

  5. 一丶人生苦短,我用python【第一篇】

    1 解释器 解释器(英语:Interpreter),又译为直译器,是一种电脑程序,能够把高级编程语言一行一行直接转译运行.解释器不会一次把整个程序转译出来,只像一位"中间人",每次 ...

  6. HDFS伪分布式环境搭建

    (一).HDFS shell操作 以上已经介绍了如何搭建伪分布式的Hadoop,既然环境已经搭建起来了,那要怎么去操作呢?这就是本节将要介绍的内容: HDFS自带有一些shell命令,通过这些命令我们 ...

  7. npm安装不成功的原因

    使用npm安装electron不成功的解决方法 使用npm安装electron不成功的解决方法 根据官网提供的electron的npm安装方法,使用下面的命令进行安装,结果不成功 npm instal ...

  8. 第18讲——string类

    关键字:string类  字符串  C-风格字符串  C库字符串函数 字符串:存储在内存的连续字节中的一系列字符. C++处理字符串的方式有两种: 来自C语言,常被称为C-风格字符串: 基于strin ...

  9. mac下管理论文的工具

    作者:丁香园站友@dlzhangyu链接:http://paper.dxy.cn/article/509726本网站所有注明“来源:丁香园”的文字.图片和音视频资料,版权均属于丁香园所有,非经授权,任 ...

  10. 编程练习:寻找发帖"水王"扩展问题二

    回顾 在前面两篇文章已经实现了水王id出现次数超过一半,以及水王id出现次数刚好一半 分析 借助上面水王id出现次数刚好出现一半的分析,其实这里就是找出数组中出现次数前三的元素,具体的分析,见前面两篇 ...