4537: [Hnoi2016]最小公倍数

题意:一张边权无向图,多组询问u和v之间有没有一条a最大为a',b最大为b'的路径(不一定是简单路径)


首先想到暴力做法,题目要求就是判断u和v连通,并查集把\(a<a' \land b<b'\)的边加入

然后想了一下特殊的莫队,不可做。不能按权值分块,因为同一个权值会有很多边,并且删除操作不好处理

发现这其实是一个偏序关系,但是无法用cdq分治,因为它要求所有满足偏序小的元素同时存在于某种组织形式中

使用分块

权值用\((a,b)\)表示

边按a排序,然后分块。对于每一块i,处理a'在这一块中的询问。这时候之前块的\(a<a'\)这一个关系一定满足,按b排序后也满足\(b<b'\)了

块i中还有一些满足的,最多\(\sqrt{m}\) 暴力加入然后撤销就可以了

不路径压缩的并查集,启发式合并的话复杂度log

注意是边权

#include <iostream>
#include <cstdio>
#include <cstring>
#include <algorithm>
#include <cmath>
using namespace std;
typedef long long ll;
const int N=2e5+5;
inline int read(){
char c=getchar();int x=0,f=1;
while(c<'0'||c>'9'){if(c=='-')f=-1;c=getchar();}
while(c>='0'&&c<='9'){x=x*10+c-'0';c=getchar();}
return x*f;
} int n, m, Q, u, v, x, y, block, bn, top, ans[N];
struct meow {
int u, v, a, b, id;
inline void print() {printf("%d %d %d %d\n", u, v, a, b);}
} e[N], q[N], a[N];
inline bool cmpa(const meow &a, const meow &b) {return a.a == b.a ? a.b < b.b : a.a < b.a;}
inline bool cmpb(const meow &a, const meow &b) {return a.b == b.b ? a.a < b.a : a.b < b.b;} namespace ufs {
struct meow{int fa, maxa, maxb, size;} t[N];
struct info{int x, y; meow a, b; } st[N];
inline int find(int x) {return x == t[x].fa ? x : find(t[x].fa);}
inline void ini() {for(int i=1; i<=n; i++) t[i] = (meow){i, -1, -1, 1}; }
inline void link(int id) { //printf("link %d\n", id);
int x = find(e[id].u), y = find(e[id].v);
st[++top] = (info){x, y, t[x], t[y]};
if(x==y) {
t[x].maxa = max(t[x].maxa, e[id].a);
t[x].maxb = max(t[x].maxb, e[id].b);
return;
}
if(t[x].size < t[y].size) swap(x, y);
t[y].fa = x;
t[x].maxa = max(max(t[y].maxa, t[x].maxa), e[id].a);
t[x].maxb = max(max(t[y].maxb, t[x].maxb), e[id].b);
t[x].size += t[y].size;
}
inline void recov() {
t[ st[top].x ] = st[top].a, t[ st[top].y ] = st[top].b;
top--;
}
inline bool check(int id) {
int x = find(a[id].u), y = find(a[id].v);
//printf("check %d %d %d %d %d %d\n", id, x, y, t[x].maxa, t[x].maxb, t[x].size);
return x == y && t[x].maxa == a[id].a && t[x].maxb == a[id].b;
}
} using ufs::st; void solve() {
sort(e+1, e+m+1, cmpa); sort(q+1, q+Q+1, cmpa);
block = sqrt(m); bn = (m-1)/block+1;
for(int i=1; i<=bn; i++) { //printf("\n---------- %d\n", i);
int l = (i-1)*block+1, r = i==bn ? m : i*block; //printf("[%d, %d]\n", l, r);
int p=0;
for(int i=1; i<=Q; i++)
if(q[i].a >= e[l].a && (r==m || q[i].a < e[r+1].a) ) a[++p] = q[i]; sort(e+1, e+l, cmpb); sort(a+1, a+p+1, cmpb);
//for(int i=1; i<=p; i++) a[i].print();
int now=1;
ufs::ini();
for(int i=1; i<=p; i++) {
while(now < l && e[now].b <= a[i].b) ufs::link(now), now++;
top=0;
for(int j=l; j<=r; j++)
if(e[j].a <= a[i].a && e[j].b <= a[i].b) ufs::link(j);// printf("j %d\n", j);
ans[ a[i].id ] = ufs::check(i);
while(top) ufs::recov();
}
}
for(int i=1; i<=Q; i++) puts(ans[i] ? "Yes" : "No");
}
int main() {
//freopen("in", "r", stdin);
freopen("multiple.in", "r", stdin);
freopen("multiple.out", "w", stdout);
n=read(); m=read();
for(int i=1; i<=m; i++) u=read(), v=read(), x=read(), y=read(), e[i] = (meow){u, v, x, y, i};
Q=read();
for(int i=1; i<=Q; i++) u=read(), v=read(), x=read(), y=read(), q[i] = (meow){u, v, x, y, i};
solve();
}

BZOJ 4537: [Hnoi2016]最小公倍数 [偏序关系 分块]的更多相关文章

  1. bzoj 4537: [Hnoi2016]最小公倍数 分块+并查集

    题目大意: 给定一张n个点m条边的无向图,每条边有两种权.每次询问某两个点之间是否存在一条路径上的边的两种权的最大值分别等于给定值. n,q <= 50000. m <= 100000 题 ...

  2. bzoj 4537 HNOI2016 最小公倍数

    Description 给定一张N个顶点M条边的无向图(顶点编号为1,2,-,n),每条边上带有权值.所有权值都可以分解成2^a*3^b的形式.现在有q个询问,每次询问给定四个参数u.v.a和b,请你 ...

  3. 4537: [Hnoi2016]最小公倍数

    Description 给定一张N个顶点M条边的无向图(顶点编号为1,2,…,n),每条边上带有权值.所有权值都可以分解成2^a*3^b的形式.现在有q个询问,每次询问给定四个参数u.v.a和b,请你 ...

  4. [BZOJ4537][HNOI2016]最小公倍数(分块+并查集)

    4537: [Hnoi2016]最小公倍数 Time Limit: 40 Sec  Memory Limit: 512 MBSubmit: 1687  Solved: 607[Submit][Stat ...

  5. [BZOJ4537][Hnoi2016]最小公倍数 奇怪的分块+可撤销并查集

    4537: [Hnoi2016]最小公倍数 Time Limit: 40 Sec  Memory Limit: 512 MBSubmit: 1474  Solved: 521[Submit][Stat ...

  6. 【BZOJ4537】[Hnoi2016]最小公倍数 分块

    [BZOJ4537][Hnoi2016]最小公倍数 Description 给定一张N个顶点M条边的无向图(顶点编号为1,2,…,n),每条边上带有权值.所有权值都可以分解成2^a*3^b的形式.现在 ...

  7. 【LG3247】[HNOI2016]最小公倍数

    [LG3247][HNOI2016]最小公倍数 题面 洛谷 题解 50pts 因为拼凑起来的部分分比较多,所以就放一起了. 以下设询问的\(a,b\)为\(A,B\), 复杂度\(O(nm)\)的:将 ...

  8. BZOJ 3343: 教主的魔法(分块+二分查找)

    BZOJ 3343: 教主的魔法(分块+二分查找) 3343: 教主的魔法 Time Limit: 10 Sec  Memory Limit: 256 MBSubmit: 1172  Solved:  ...

  9. [BZOJ 3509] [CodeChef] COUNTARI (FFT+分块)

    [BZOJ 3509] [CodeChef] COUNTARI (FFT+分块) 题面 给出一个长度为n的数组,问有多少三元组\((i,j,k)\)满足\(i<j<k,a_j-a_i=a_ ...

随机推荐

  1. 使用 SVG 和 JS 创建一个由星形变心形的动画

    序言:首先,这是一篇学习 SVG 及 JS 动画不可多得的优秀文章.我非常喜欢 Ana Tudor 写的教程.在她的教程中有大量使用 SVG 制作的图解以及实时交互 DEMO,可以说教程的所有细枝末节 ...

  2. Generator函数异步应用

    转载请注明出处: Generator函数异步应用 上一篇文章详细的介绍了Generator函数的语法,这篇文章来说一下如何使用Generator函数来实现异步编程. 或许用Generator函数来实现 ...

  3. [国嵌笔记][007][Linux网络配置]

    Vmware网络设置 1.bridged(桥接模式) 如果网络中能提供多个IP地址,则使用桥接方式.虚拟机与主机的IP地址彼此独立. 2.NAT(网络地址转换模式) 如果只能提供一个IP地址,则使用N ...

  4. 如何开发由Create-React-App 引导的应用(一)

    此文章是翻译How to develop apps bootstrapped with Create React App 官方文档 系列文章 如何开发由Create-React-App 引导的应用 如 ...

  5. 详解 Vue 2.4.0 带来的 4 个重大变化

    在这篇文章中,我将跟大家分享4个有突破性新特性. 服务端渲染异步组件 包裹组件内实现属性继承 异步组件支持webpack3 组件渲染后可保留HTML注释 1.服务端渲染异步组件 在vue2.4.0以前 ...

  6. thinkphp 中concat(连接)使用方法

    1.concat将title和id连接作为truename新的字段,left从url字段左侧开始截取25个字符,同理right也可. 2.getLastql用法:

  7. 一步步部署基于Windows系统的Jenkins持续集成环境

    如题:本文将介绍如何在Windows环境下运用Jenkins部署持续集成环境.之所以写本文,是因为在最近工作当中,学习使用Jenkins时,确实遇到了一些问题,而大多数教程文档都是基于Mac或是Lin ...

  8. OpenGL+OpenCV实现立方体贴图

    我屮艸芔茻,转眼就7月份了. 今天试了一下立方体贴图,比较简单,大概说下和平面贴图的区别. 1. 平面贴图需要的是纹理坐标vec2:立方体贴图需要的是一个方向向量vec3,长度没有关系,重要的是方向, ...

  9. “Project 'MyFunProject' is not a J2SE 5.0 compliant project.”

  10. CCF系列之出现次数最多的数(201312-1)

    试题名称: 出现次数最多的数 时间限制: 1.0s 内存限制: 256.0MB 问题描述: 问题描述 给定n个正整数,找出它们中出现次数最多的数.如果这样的数有多个,请输出其中最小的一个.   输入格 ...