题目链接

LOJ:https://loj.ac/problem/3057

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

Solution

先膜一发\(myy\),这里是官方题解

对于\(30\)分,可以得到一个很显然的做法:设\(f[i][j]\)表示\(i\to j\)的答案,然后枚举\(i,j\)的出边转移,这样做是\(O(m^2)\)的。

对于\(100\)分,注意到点的规模没变,我们有一个缩边的想法。

我们把边分成三类,同色的两种和异色的一种,对每类分别建图,然后我们对每个联通块进行考虑:

  • 如果当前联通块是二分图,那么可以说明从\(i\)点到\(j\)点所经过的边数必然只能是奇数或偶数,那么我们任意处理出一棵生成树,容易发现这棵生成树和原图等价(如果这边多走了就在另一边来回走以抵消)。
  • 如果当前联通块不是二分图,那么边数可以是奇数或偶数,那么我们只需要在生成树任意加一个自环就好了,这样的话就可以满足边数任意奇偶,注意异色边建出来的图不可能有这种情况。

缩完之后可以发现边数是\(O(n)\)级别的,那么我们套用上边的暴力,复杂度达到了\(O(n^2)\),足已通过此题。

#include<bits/stdc++.h>
using namespace std; void read(int &x) {
x=0;int f=1;char ch=getchar();
for(;!isdigit(ch);ch=getchar()) if(ch=='-') f=-f;
for(;isdigit(ch);ch=getchar()) x=x*10+ch-'0';x*=f;
} void print(int x) {
if(x<0) putchar('-'),x=-x;
if(!x) return ;print(x/10),putchar(x%10+48);
}
void write(int x) {if(!x) putchar('0');else print(x);putchar('\n');} #define lf double
#define ll long long const int maxn = 2e5+10;
const int inf = 1e9;
const lf eps = 1e-8; char s[maxn];
int n,m,Q,vis[maxn],sta[maxn],top,f[5050][5050],flag; struct Graph {
int head[maxn],tot,fa[maxn];
struct edge{int to,nxt;}e[maxn*20]; void add(int u,int v) {e[++tot]=(edge){v,head[u]},head[u]=tot;}
void ins(int u,int v) {add(u,v),add(v,u);} void dfs(int x,int c) {
vis[x]=c,sta[++top]=x;
for(int i=head[x];i;i=e[i].nxt)
if(!vis[e[i].to]) dfs(e[i].to,3-c),fa[e[i].to]=x;
else flag|=(vis[e[i].to]!=3-c);
} void solve() {
queue<pair<int,int > > q;
for(int i=1;i<=n;i++) f[i][i]=1,q.push(make_pair(i,i));
for(int x=1;x<=n;x++)
for(int i=head[x];i;i=e[i].nxt)
if(s[x]==s[e[i].to]) f[x][e[i].to]=1,q.push(make_pair(x,e[i].to));
while(!q.empty()) {
int x=q.front().first,y=q.front().second;q.pop();
for(int i=head[x];i;i=e[i].nxt)
for(int u,v,j=head[y];j;j=e[j].nxt)
if(s[u=e[i].to]==s[v=e[j].to])
if(!f[u][v]) f[u][v]=1,q.push(make_pair(u,v));
}
}
}g[3],t; int main() {
read(n),read(m),read(Q);scanf("%s",s+1);
for(int i=1,x,y;i<=m;i++) {
read(x),read(y);
if(s[x]!=s[y]) g[0].ins(x,y);
else if(s[x]=='1') g[1].ins(x,y);
else g[2].ins(x,y);
}
for(int i=0;i<=2;i++) {
memset(vis,0,sizeof vis);
for(int j=1;j<=n;j++)
if(!vis[j]) {
top=flag=0,g[i].dfs(j,1);
while(top) {
int x=sta[top];
if(g[i].fa[x]) t.ins(g[i].fa[x],x);top--;
}if(flag) t.ins(j,j);
}
}
t.solve();
for(int i=1,x,y;i<=Q;i++) read(x),read(y),puts(f[x][y]?"YES":"NO");
return 0;
}

[LOJ3057] [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. [HNOI2019]校园旅行

    题意 https://www.luogu.org/problemnew/show/P5292 思考 最朴素的想法,从可行的二元组(u,v)向外拓展,及u的出边所指的颜色与v的出边所指的颜色若相同,继续 ...

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

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

  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. MySQL高级-慢查询日志

    一.慢查询日志是什么 1. 2. 3. 2.开启了慢查询日志后,什么样的SQL才会记录到慢查询日志里面呢? 3.案例 1.查看当前多少秒算慢 2.设置慢的阙值时间 3.为什么设置后看不出变化? 4.记 ...

  2. 请求头(request)和响应头(response)

    说一说常见的请求头和相应头都有什么呢? 1)请求(客户端->服务端[request]) GET(请求的方式) /newcoder/hello.html(请求的目标资源) HTTP/1.1(请求采 ...

  3. ionic ios样式偏移解决方案。

    在css属性内增加: .item-ios [item-end] { //解决ios系统上尾部图标出现重影而增加的格式. margin: 0px -15.3px 0px 0px; margin-bott ...

  4. 从零开始的Python学习Episode 3——字符串格式化与for循环

    一.字符串格式化 利用一段注释记录想要输出的字符串格式,并用 %s . %d 或 %f 依次代替要输出的数据(%s代表字符串,%d代表数字,%f代表浮点数),然后在这段注释之后依次加上要输出的数据. ...

  5. 动画效果 ObjectAnimator

    学习了一下动画效果的使用,做一下笔记 ImageView imageView = findViewById(R.id.imageView); ObjectAnimator.ofFloat(imageV ...

  6. 如何在etherscan提交代币官方信息

    https://ethlinkersupport.zendesk.com/hc/zh-cn/articles/360001334992-%E5%A6%82%E4%BD%95%E5%9C%A8ether ...

  7. ThinkPHP - 1 - 本地部署

    ThinkPHP ThinkPHP是一个快速.简单的基于MVC和面向对象的轻量级PHP开发框架,遵循Apache2开源协议发布,从诞生以来一直秉承简洁实用的设计原则,在保持出色的性能和至简的代码的同时 ...

  8. Image控件显示以byte[]字节数组形式存在的图片

    工作中遇到了这样的一个问题.起初觉得很简单,获得了图片的byte[]后,可以将其转换成内存中的图片对象(如System.Drawing.Image),而后赋给页面的Image控件.尝试后才发现这样根本 ...

  9. quartz入门实例

    一 Quarta介绍 1 Quartz是什么 Quartz就是一个纯 Java 实现的作业调度工具,相当于数据库中的 Job.Windows 的计划任务.Unix/Linux 下的 Cron,但 Qu ...

  10. Python—集合(在我的世界,你就是唯一)

    一.概念与定义 集合类型与数学中集合的概念一致,即包含0个或多个数据项的无序组合. 元素不可重复,只能是固定数据类型元素. 集合(set)属于Python无序可变序列,使用一对大括号作为定界符,元素之 ...