传送门

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

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

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

首先求链的公共部分,设两种蚂蚁为 $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. 通过 PHP 生成 XML

    如需使用 PHP 在服务器上生成 XML 响应,请使用下面的代码: <?php header("Content-type:text/xml"); echo "< ...

  2. 观察者模式------《Head First 设计模式》

    第二章---观察者模式 xzmxddx 学习方式:书籍<Head First 设计模式>,这本书通俗易懂,所有知识点全部取自本书. 面向对象设计原则 封装变化 多用组合,少用继承 针对接口 ...

  3. java+struts上传文件夹文件

    这里只写后端的代码,基本的思想就是,前端将文件分片,然后每次访问上传接口的时候,向后端传入参数:当前为第几块文件,和分片总数 下面直接贴代码吧,一些难懂的我大部分都加上注释了: 上传文件实体类: 看得 ...

  4. BZOJ 4769: 超级贞鱼 逆序对 + 归并排序

    手画几下序列的变换后发现逆序对数是恒定的,故只需对第 $0$ 年求逆序对即可. 树状数组会 $TLE$ 的很惨,需要用到归并排序来求逆序对. 其实就是省掉了一个离散化的时间,估计能比树状数组快一半的时 ...

  5. Elastic-Job介绍

    1 什么是分布式任务调度 什么是分布式?当前软件的架构正在逐步转变为分布式架构,将单体结构分为若干服务,服务之间通过网络交互来完成用户的业务处理,如下图,电商系统为分布式架构,由订单服务.商品服务.用 ...

  6. 序列式容器————list

    list是一个线性双向链表结构,它的数据由若干个节点构成,每一个节点都包括一个信息块(即实际存储的数据).一个前驱指针和一个后驱指针. 它无需分配指定的内存大小且可以任意伸缩,这是因为它存储在非连续的 ...

  7. vscode Vue格式化HTML标签换行问题

    解决方法:在设置里面直接贴上代码: "vetur.format.defaultFormatter.html": "js-beautify-html", &quo ...

  8. 为什么JPA@Modifying需要@Transactional注解

    在JPA开发中遇到一个很奇怪的问题,@Modifying需要和@Transactional配合使用才能正常使用.如下面代码所示 @Modifying @Transactional @Query(&qu ...

  9. js 放回上一页

    window.history.go(-1);//返回上一页不刷新 window.location.href = document.referrer;//返回上一页并刷新

  10. js fuction函数内return一个内部函数详解

    今天在网上,看到一篇关于js函数难点的文章,js函数的一些难点.在那上面提了一下,关于js函数返回另一个函数的问题,并附上了一道面试题: var add = function(x){ var sum  ...