题意可以转化为是否能找一条从\(u\)到\(v\)的路径,经过的边的\(a\)和\(b\)的最大值恰好都是询问所给定的值。

若只有\(a\)的限制,可以将询问离线,对边和询问都从小到大排序,然后双指针维护当前合法的边,用并查集维护连通块的最值和连通性。

现在有\(a\)和\(b\)的限制,考虑对边分块,先对所有边按\(a\)从小到大排序,对所有询问按\(b\)从小到大排序。

考虑当前枚举到的一个块及其之前块对询问的贡献。对所有询问找到\(a\)大小恰好在当前块范围内的询问,对当前块之前的整块按\(b\)从小到大排序,这样就能保证在当前块之前的所有边的\(a\)都小于等于现在找到的这些询问,然后这些边对\(b\)都有序,因为已经事先已经对询问排过序,所以可以像只有一个限制时一样,对当前块之前的整块进行双指针维护对当前询问合法的边。

对于零散块内的边,可以直接暴力枚举,如果合法就加入,当考虑到下一个询问时,对于零散块内加入的边要执行删除,所以用可撤销并查集来维护即可。

\(code:\)

#include<bits/stdc++.h>
#define maxn 200010
using namespace std;
template<typename T> inline void read(T &x)
{
x=0;char c=getchar();bool flag=false;
while(!isdigit(c)){if(c=='-')flag=true;c=getchar();}
while(isdigit(c)){x=(x<<1)+(x<<3)+(c^48);c=getchar();}
if(flag)x=-x;
}
int n,m,qu,S,top;
int ans[maxn],fa[maxn],de[maxn],va[maxn],vb[maxn];
struct node
{
int x,y,a,b,id;
}e[maxn],q[maxn],t[maxn];
bool cmp1(const node &x,const node &y)
{
if(x.a==y.a) return x.b<y.b;
return x.a<y.a;
}
bool cmp2(const node &x,const node &y)
{
if(x.b==y.b) return x.a<y.a;
return x.b<y.b;
}
struct Node
{
int x,y,deep,a,b;
}st[maxn];
int find(int x)
{
return fa[x]==x?x:find(fa[x]);
}
void merge(int x,int y,int a,int b)
{
x=find(x),y=find(y);
if(de[x]<de[y]) swap(x,y);
st[++top]=(Node){x,y,de[x],va[x],vb[x]};
va[x]=max(va[x],a),vb[x]=max(vb[x],b);
if(x==y) return;
fa[y]=x,de[x]=max(de[x],de[y]+1);
va[x]=max(va[x],va[y]),vb[x]=max(vb[x],vb[y]);
}
bool query(int id)
{
int x=find(t[id].x),y=find(t[id].y);
if(x!=y) return false;
if(va[x]!=t[id].a||vb[x]!=t[id].b) return false;
return true;
}
void del(int id)
{
int x=st[id].x,y=st[id].y;
fa[y]=y,de[x]=st[id].deep,va[x]=st[id].a,vb[x]=st[id].b;
}
int main()
{
read(n),read(m),S=sqrt(m*log2(n));
for(int i=1;i<=m;++i)
read(e[i].x),read(e[i].y),read(e[i].a),read(e[i].b);
read(qu);
for(int i=1;i<=qu;++i)
read(q[i].x),read(q[i].y),read(q[i].a),read(q[i].b),q[i].id=i;
sort(e+1,e+m+1,cmp1),sort(q+1,q+qu+1,cmp2);
for(int i=1;i<=m;i+=S)
{
int tot=0,pos=1;
for(int j=1;j<=n;++j) fa[j]=j,va[j]=vb[j]=-1,de[j]=0;
for(int j=1;j<=qu;++j)
if(q[j].a>=e[i].a&&(q[j].a<e[i+S].a||i+S>m))
t[++tot]=q[j];
if(!tot) continue;
if(i!=1) sort(e+1,e+i,cmp2);
for(int j=1;j<=tot;++j)
{
while(pos<i&&e[pos].b<=t[j].b)
merge(e[pos].x,e[pos].y,e[pos].a,e[pos].b),pos++;
top=0;
for(int k=i;k<i+S&&k<=m;++k)
if(e[k].a<=t[j].a&&e[k].b<=t[j].b)
merge(e[k].x,e[k].y,e[k].a,e[k].b);
ans[t[j].id]=query(j);
while(top) del(top--);
}
}
for(int i=1;i<=qu;++i)
{
if(ans[i]) puts("Yes");
else puts("No");
}
return 0;
}

最新文章

  1. PHP系统声明式事务处理
  2. html5 离线存储
  3. poj 1730 Perfect Pth Powers
  4. 使用fastjson前台报406的问题解决方法
  5. 【动态规划】Codeforces 698A &amp; 699C Vacations
  6. [读书笔记] 四、SpringBoot中使用JPA 进行快速CRUD操作
  7. java项目和java-web项目中文件和文件夹的含义
  8. [SDOI2006]仓库管理员的烦恼
  9. SpringBoot与日志框架1(基本使用)
  10. dajngo cache,throttling
  11. Eureka源码解读
  12. 2050 Programming Competition
  13. java Page分页显示
  14. cocos2d-x游戏引擎核心(3.x)----事件分发机制之事件从(android,ios,desktop)系统传到cocos2dx的过程浅析
  15. quartz---springmvc的配置文件正合
  16. 安装Python 3.6
  17. Burp 之Intruder
  18. CONVERT函数----SQL
  19. iOS:Xcode7下创建 .a静态库 和 .framework静态库
  20. 二叉查找树--java

热门文章

  1. [ 头皮发麻 A1 ] 队内赛3 2020 Ateneo de Manila University DISCS PrO HS Division
  2. java 中对hashmap进行排序
  3. SqlServer2016 startengine错误的解决方式整理
  4. 【neo4j】文件管理路径、数据备份、创建新数据库、导入数据等操作记录
  5. 表达式计算开源组件(NCalc.NetCore)
  6. Glusterfs的安装、创建卷、配置和优化卷、挂载使用
  7. 如何用好 IDEA ,Java 撸码效率至少提升 5 倍?
  8. Yolo训练自定义目标检测
  9. html中绝对路径和相对路径的区别?比较相对路径和绝对路径的优缺点
  10. Monkey and Banana 题解(动态规划)