题意

https://www.luogu.org/problemnew/show/P5292


思考

最朴素的想法,从可行的二元组(u,v)向外拓展,及u的出边所指的颜色与v的出边所指的颜色若相同,继续更新二元组(u',v'),复杂度约为O(m2)。

我们发现,很多时候边上的转移其实是没有必要的,因为有很多情况能转移到相同的字符串,因此我们要删去一些边,并且使得不改变原图的性质。

先考虑原图中边的两端颜色相同的边构成的连通块,若为二分图,则取其任意生成树;若不为二分图,则取其任意生成树,并添加一个自环。

二分图的意思,就是相同颜色段的长度只与奇偶性有关,当字符串另一端不断添加相同字符的同时,二分图中的字符可以不断地在A集合与B集合中转换。

两端颜色不同的边构成的连通块,也是取其任意生成树。

这样边数为O(n)级别,复杂度为O(n2)。


代码

 // luogu-judger-enable-o2
#include<bits/stdc++.h>
using namespace std;
const int maxn=5E5+;
int head[maxn*],size,n,m,x,y,color[maxn],flag,q;
char c[maxn];
bool f[][],can[maxn];
struct edge{int from,to,next;};
struct pt{int x,y;};
queue<pt>Q;
struct graph
{
edge E[maxn*];
int head[maxn*],size;
void add(int u,int v)
{
E[++size].to=v;
E[size].next=head[u];
E[size].from=u;
head[u]=size;
}
}A,B,G;
void dfs1(int u,int now)
{
color[u]=now;
for(int i=A.head[u];i;i=A.E[i].next)
{
int v=A.E[i].to;
if(now==color[v])flag=;
if(color[v])continue;
G.add(u,v);
G.add(v,u);
if(now==)dfs1(v,);
else dfs1(v,);
}
}
void dfs2(int u)
{
color[u]=;
for(int i=B.head[u];i;i=B.E[i].next)
{
int v=B.E[i].to;
if(color[v])continue;
G.add(u,v);
G.add(v,u);
dfs2(v);
}
}
void init1()
{
for(int i=;i<=A.size;++i)
can[A.E[i].from]=can[A.E[i].to]=;
for(int u=;u<=n;++u)
{
if(color[u]||!can[u])continue;
flag=;
dfs1(u,);
if(flag)G.add(u,u);
}
}
void init2()
{
memset(color,,sizeof(color));
memset(can,,sizeof(can));
for(int i=;i<=B.size;++i)
can[B.E[i].from]=can[B.E[i].to]=;
for(int u=;u<=n;++u)
{
if(color[u]||!can[u])continue;
dfs2(u);
}
}
void work()
{
while(!Q.empty())
{
pt u=Q.front();
Q.pop();
for(int i=G.head[u.x];i;i=G.E[i].next)
{
for(int j=G.head[u.y];j;j=G.E[j].next)
{
int u=G.E[i].to,v=G.E[j].to;
if(c[u]==c[v])
{
if(!f[u][v])Q.push((pt){u,v});
f[u][v]=f[v][u]=;
}
}
}
}
}
int main()
{
ios::sync_with_stdio(false);
cin>>n>>m>>q;
for(int i=;i<=n;++i)
{
cin>>c[i];
f[i][i]=;
Q.push((pt){i,i});
}
for(int i=;i<=m;++i)
{
cin>>x>>y;
if(c[x]==c[y])
{
A.add(x,y),A.add(y,x);
f[x][y]=f[y][x]=;
Q.push((pt){x,y});
}
else B.add(x,y),B.add(y,x);
}
init1();
init2();
work();
for(int i=;i<=q;++i)
{
cin>>x>>y;
if(f[x][y])cout<<"YES"<<endl;
else cout<<"NO"<<endl;
}
return ;
}

[HNOI2019]校园旅行的更多相关文章

  1. 【BZOJ5492】[HNOI2019]校园旅行(bfs)

    [HNOI2019]校园旅行(bfs) 题面 洛谷 题解 首先考虑暴力做法怎么做. 把所有可行的二元组全部丢进队列里,每次两个点分别向两侧拓展一个同色点,然后更新可行的情况. 这样子的复杂度是\(O( ...

  2. [HNOI2019]校园旅行(构造+生成树+动规)

    题目 [HNOI2019]校园旅行 做法 最朴素的做法就是点对扩展\(O(m^2)\) 发现\(n\)比较小,我们是否能从\(n\)下手减少边数呢?是肯定的 单独看一个颜色的联通块,如果是二分图,我们 ...

  3. Luogu P5292 [HNOI2019]校园旅行

    非常妙的一道思博题啊,不愧是myy出的题 首先我们考虑一个暴力DP,直接开一个数组\(f_{i,j}\)表示\(i\to j\)的路径能否构成回文串 考虑直接拿一个队列来转移,队列里存的都是\(f_{ ...

  4. 洛谷P5292 [HNOI2019]校园旅行(二分图+最短路)

    题面 传送门 题解 如果暴力的话,我们可以把所有的二元组全都扔进一个队列里,然后每次往两边更新同色点,这样的话复杂度是\(O(m^2)\) 怎么优化呢? 对于一个同色联通块,如果它是一个二分图,我们只 ...

  5. [LOJ3057] [HNOI2019] 校园旅行

    题目链接 LOJ:https://loj.ac/problem/3057 洛谷:https://www.luogu.org/problemnew/show/P5292 Solution 先膜一发\(m ...

  6. 【洛谷5292】[HNOI2019] 校园旅行(思维DP)

    点此看题面 大致题意: 给你一张无向图,每个点权值为\(0\)或\(1\),多组询问两点之间是否存在一条回文路径. 暴力\(DP\) 首先,看到\(n\)如此之小(\(n\le5000\)),便容易想 ...

  7. bzoj5492:[Hnoi2019]校园旅行

    传送门 %%%myy 考虑30分做法:暴力bfs,\(f[i][j]\)表示\(i\)到\(j\)可以形成回文串 然而为什么我场上只想到了70分做法,完全没想到30分怎么写.. 100分: 考虑缩边, ...

  8. [HNOI2019]校园旅行(建图优化+bfs)

    30分的O(m^2)做法应该比较容易想到:令f[i][j]表示i->j是否有解,然后把每个路径点数不超过2的有解状态(u,v)加入队列,然后弹出队列时,两点分别向两边搜索边,发现颜色一样时,再修 ...

  9. Loj #3057. 「HNOI2019」校园旅行

    Loj #3057. 「HNOI2019」校园旅行 某学校的每个建筑都有一个独特的编号.一天你在校园里无聊,决定在校园内随意地漫步. 你已经在校园里呆过一段时间,对校园内每个建筑的编号非常熟悉,于是你 ...

随机推荐

  1. PL/SQL变量和类型

    变量 在定义变量时一定要为其指定一个类型,类型可以是PL/SQL类型或SQL语言的类型,一旦变量的类型确定,那么变量中所能存储的值也就确定了,因此尽管变量的值会经常改变,但是值的类型是不可以变化的. ...

  2. UI自动化(十三)appium操作

    定位: # id resource-id可能出现重复 如果appium-desktop提示了id 定位方式 说明id唯一driver.find_element_by_id('com.tencent.m ...

  3. bzoj 4770 图样 - 概率与期望 - 动态规划

    题目传送门 传送门I 传送门II 题目大意 有一个$n$个点的完全图,每个点的权值是$[0, 2^{m})$中的随机整数,两点间的边的权值是两点点权的异或和,问它的最小异或生成树的边权和的期望. 考虑 ...

  4. 2018年-2019年第二学期第七周C#学习个人总结

    在本周我又学习了,第五章面向对象高级中的5.5异常和5.6命名空间和程序集.在5.5异常中我知道了异常层次结构:所有的异常类都继承自Exception类.由于发生了异常程序立即终止无法再继续向下执行. ...

  5. Oh-My-Zsh及主题、插件安装与配置

    切换zsh Manjaro linux默认安装了zsh,其他可能需要先安装 cat /etc/shells #查看本地有哪几种shell chsh -s /bin/zsh #切换到zsh 默认终端启动 ...

  6. bzoj3527: [Zjoi2014]力 fft

    bzoj3527: [Zjoi2014]力 fft 链接 bzoj 思路 但是我们求得是 \(\sum\limits _{i<j} \frac{q_i}{(i-j)^2}-\sum_{i> ...

  7. P4389 付公主的背包

    注意 初始化的时候要这样写 for(int i=1,x;i<=n;i++){ scanf("%d",&x); v[x]++; } for(int i=1;i<= ...

  8. JAVA程序错误总结

    NoSuchMethodError:main 出错原因:找不到main方法,应该是main方法书写有误. 解决方案:检查public static void main(String [] args){ ...

  9. 删除List集合中的元素方法

    List集合是我们平时使用的最多的集合了,一般用来存放从数据库中查询的对象数据,但有时我们会从中筛选不需要的数据,第一次使用这种方式: 使用增强for循环遍历,使用list的remove方法删除不符合 ...

  10. Scss 与 Sass 是什么,他们的区别在哪里?

    转载自:http://yunkus.com/difference-between-scss-sass/ 要想了解Scss 与 Sass 是什么以及他们的区别又在哪里,我们不过不先从他们各自的定义说起. ...