传送门

发现蚂蚁不多,所以考虑两两枚举然后判断

那么首先要求出两条链的公共部分,然后根据之间在公共链的时间段和是同向还是反向进行判断

思路简单但是细节很多......

首先求链的公共部分,设两种蚂蚁为 $a,b$,路径分别为 $As,At$,$Bs,Bt$

那么经过一波手玩分类讨论,公共部分的两端点就是 $LCA(As,Bs),LCA(As,Bt),LCA(At,Bs),LCA(At,Bt)$ 之间 $dfs$ 序较大的两个

记得之前要先把不在两条链上的点去掉,然后如果最后只有一个点合法则公共部分只有一个点

然后每只蚂蚁在公共链的起点就是距离比较近的那个,判断是否同向只要看看公共链起点是否相等即可

判断点是否在某条链上,因为边权不为 $0$ ,所以可以用点到链两端点的距离和是否等于链长来判断

然后如果两只蚂蚁同向,那么只要判断先到达公共链起点的后到达公共链终点即可

如果不同向,那么要判断它们在公共链的时间区间是否有交

为了避免除法所以移项变成乘法,注意乘会爆 $int$

因为多次求 $LCA$,所以要 $RMQ$ $O(1)$ 求 $LCA$

具体实现我感觉自己代码还挺好看的...

#include<iostream>
#include<cstdio>
#include<algorithm>
#include<cstring>
#include<cmath>
using namespace std;
typedef long long ll;
inline int read()
{
int x=,f=; char ch=getchar();
while(ch<''||ch>'') { if(ch=='-') f=-; ch=getchar(); }
while(ch>=''&&ch<='') { x=(x<<)+(x<<)+(ch^); ch=getchar(); }
return x*f;
}
const int N=2e5+,INF=1e9+;
int n,m,S[N],T[N],V[N],Ans;
int fir[N],from[N<<],to[N<<],val[N<<],cntt;
inline void add(int a,int b,int c)
{
from[++cntt]=fir[a]; fir[a]=cntt;
to[cntt]=b; val[cntt]=c;
}
int dep[N],dfn[N],cnt,st[N<<],pos[N<<],Top;
void dfs(int x,int fa)
{
dfn[x]=++cnt; st[++Top]=x; pos[x]=Top;
for(int i=fir[x];i;i=from[i])
{
int &v=to[i]; if(v==fa) continue;
dep[v]=dep[x]+val[i]; dfs(v,x); st[++Top]=x;
}
}
int F[N<<][],Log[N<<];
void pre()
{
Log[]=-; for(int i=;i<=Top;i++) Log[i]=Log[i>>]+;//
for(int i=;i<=Top;i++) F[i][]=st[i];
for(int i=;(<<i)<=Top;i++)
for(int j=;j+(<<i-)<=Top;j++)
{
if( dfn[F[j][i-]] < dfn[ F[j+(<<i-)][i-] ] ) F[j][i]=F[j][i-];
else F[j][i]=F[j+(<<i-)][i-];
}
}
inline int query(int x,int y)
{
int l=pos[x],r=pos[y],k; if(l>r) swap(l,r);
k=Log[r-l+];
if(dfn[F[l][k]]<dfn[F[r-(<<k)+][k]]) return F[l][k];
else return F[r-(<<k)+][k];
}
inline ll dis(int x,int y) { return dep[x]+dep[y]-*dep[query(x,y)]; }
inline bool cmp(const int &A,const int &B) { return dfn[A]>dfn[B]; }
inline bool in_chain(int x,int y,int a) { return dis(x,a)+dis(y,a)==dis(x,y); }
inline bool pd(ll l,ll r,ll x) { return x>=l && x<=r; }
inline bool solve(int As,int At,int Bs,int Bt,int Av,int Bv)
{
int x[],tot=,p=query(As,Bs);
if(in_chain(As,At,p)&&in_chain(Bs,Bt,p)) x[++tot]=p;
p=query(As,Bt);
if(in_chain(As,At,p)&&in_chain(Bs,Bt,p)) x[++tot]=p;
p=query(At,Bs);
if(in_chain(As,At,p)&&in_chain(Bs,Bt,p)) x[++tot]=p;
p=query(At,Bt);
if(in_chain(As,At,p)&&in_chain(Bs,Bt,p)) x[++tot]=p;
if(!tot) return ;
sort(x+,x+tot+,cmp); if(tot==) x[]=x[];
int Al=(dis(As,x[])<dis(As,x[]) ? x[] : x[]),Ar=(Al==x[] ? x[] : x[]);
int Bl=(dis(Bs,x[])<dis(Bs,x[]) ? x[] : x[]),Br=(Bl==x[] ? x[] : x[]);
ll Asl=dis(As,Al)*Bv,Bsl=dis(Bs,Bl)*Av,Asr=dis(As,Ar)*Bv,Bsr=dis(Bs,Br)*Av;
if(Al==Bl)
{
if(Asl<=Bsl && Asr>=Bsr) return ;
if(Asl>=Bsl && Asr<=Bsr) return ;
return ;
}
if(pd(Asl,Asr,Bsl)||pd(Asl,Asr,Bsr)) return ;
if(pd(Bsl,Bsr,Asl)||pd(Bsl,Bsr,Asr)) return ;
return ;
}
int main()
{
int TT=read();
while(TT--)
{
cntt=Top=cnt=Ans=; for(int i=;i<=n;i++) fir[i]=;
n=read(); int a,b,c;
for(int i=;i<n;i++)
{
a=read(),b=read(),c=read();
add(a,b,c); add(b,a,c);
}
m=read();
for(int i=;i<=m;i++) S[i]=read(),T[i]=read(),V[i]=read();
dfs(,); pre();
for(int i=;i<=m;i++)
for(int j=i+;j<=m;j++)
if(solve(S[i],T[i],S[j],T[j],V[i],V[j])) Ans++;
printf("%d\n",Ans);
}
return ;
}

BZOJ 1906. 树上的蚂蚁的更多相关文章

  1. BZOJ.3784.树上的路径(点分治 贪心 堆)

    BZOJ \(Description\) 给定一棵\(n\)个点的带权树,求树上\(\frac{n\times(n-1)}{2}\)条路径中,长度最大的\(m\)条路径的长度. \(n\leq5000 ...

  2. BZOJ 3784: 树上的路径

    Description 问一棵树上前 \(k\) 大路径的边权. Sol 边分治. 非常感谢数据没有菊花图. 为了写写边分治试试然后就开了这道题. 边分治非常好想,选一条重边,分成两部分,然后分别求最 ...

  3. bzoj 3784: 树上的路径 堆维护第k大

    3784: 树上的路径 Time Limit: 10 Sec  Memory Limit: 256 MBSubmit: 88  Solved: 27[Submit][Status][Discuss] ...

  4. [BZOJ 1033] [ZJOI2008] 杀蚂蚁antbuster 【模拟!】

    题目链接: BZOJ - 1033 题目分析 模拟!纯粹按照题目描述模拟! 这是一道喜闻乐见的经典模拟题! 我一共写了2遍,Debug 历时2天的所有晚自习 ... 时间超过 8h ... 我真是太弱 ...

  5. BZOJ 1316: 树上的询问( 点分治 + 平衡树 )

    直接点分治, 用平衡树(set就行了...)维护. -------------------------------------------------------------------------- ...

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

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

  7. bzoj 4033 树上染色 - 树形动态规划

    有一棵点数为N的树,树边有边权.给你一个在0~N之内的正整数K,你要在这棵树中选择K个点,将其染成黑 色,并将其他的N-K个点染成白色.将所有点染色后,你会获得黑点两两之间的距离加上白点两两之间距离的 ...

  8. [BZOJ 1033][ZJOI2008]杀蚂蚁antbuster

    1033: [ZJOI2008]杀蚂蚁antbuster Time Limit: 10 Sec  Memory Limit: 128 MBSubmit: 1200  Solved: 507[Submi ...

  9. bzoj 3757 树上莫队

    感谢以下文章作者: http://blog.csdn.net/kuribohg/article/details/41458639 http://vfleaking.blog.163.com/blog/ ...

随机推荐

  1. python调用Opencv库和dlib库

    python是一门胶水语言,可以调用C++编译好的dll库 python调用opencv-imggui.dll文件 https://www.cnblogs.com/zhangxian/articles ...

  2. FP回写阶段卡住或报错

    FP在autosap阶段往往会因各种各样的因素导致回写报错,卡住等异常: 本次由于一个视图的性能突然下降导致回写长期卡住没有完成: 这时如果要重新执行回写的话就要检查哪部分的数据回写到SAP端,一般有 ...

  3. Proxy-代理器(预计vue3.0实现双向绑定的方式)

    todo 常见的基于数据劫持的双向绑定有两种实现,一个是目前Vue在用的Object.defineProperty,另一个是ES2015中新增的Proxy,而Vue的作者宣称将在Vue3.0版本后加入 ...

  4. wannafly 练习赛11 B 假的字符串(字典树+建边找环)

    链接:https://www.nowcoder.com/acm/contest/59/B 时间限制:C/C++ 1秒,其他语言2秒 空间限制:C/C++ 32768K,其他语言65536K 64bit ...

  5. jQuery-validate插件初级篇

    特别提示:本人博客部分有参考网络其他博客,但均是本人亲手编写过并验证通过.如发现博客有错误,请及时提出以免误导其他人,谢谢!欢迎转载,但记得标明文章出处:http://www.cnblogs.com/ ...

  6. 后盾网lavarel视频项目---页面post方式提交之后动态弹出错误信息

    后盾网lavarel视频项目---页面post方式提交之后动态弹出错误信息 一.总结 一句话总结: 1.思路和我想的一样,有错误的时候弹出提示错误消息的模态框就好,没有错误的时候不管它 2.把模态框的 ...

  7. Promethus

    https://blog.csdn.net/zl1zl2zl3/article/details/74332437

  8. HTTPS 证书制作及使用

    一 证书的制作 进入jdk/bin,使用keytools.exe制作证书. 1.创建keystore 创建一个别名为serverkeystore的证书,该证书存放在名为server.keystore的 ...

  9. Linux_文件系统、磁盘分区_RHEL7

    目录 目录 前言 文件系统 目录结构 文件的类型 文件系统损坏后的修复 磁盘分区 分区的类型 分区最小存储单元 查看当前分区的block的大小 分区格式 MBR格式 GPT格式 mount挂载指令 挂 ...

  10. 你知道 Git 是如何做版本控制的吗?(转)

    总结:阅读这篇文章需要20分钟 本文是转载自 滴滴WebApp架构组 的一篇文章,文章讲解了神秘的.git目录下的一些文件,最终阐述了git是如何存储数据,及git分支的相关内容. git如何存储数据 ...