题解 CSP2019-J2T4【加工零件】
这题我们要求的是啥呢?仔细读题可以发现,工人传送带的关系可以看成一个 \(n\) 个点和 \(m\) 条边的无向图,然后对于每组询问 \((a,L)\),其实就是问:
\(1\) 到 \(a\) 有没有一条长度为 \(L\) 的路径。
我们换个角度思考一下,如果已知 \(1\) 到 \(a\) 有一条长度为 \(S\) 的路径,我们在这条路径上任选一条边重复走一次,那么就会出现一条 \(1\) 到 \(a\) 长度为 \(S+2\)的路径 ,同理,也会有 \(1\) 到 \(a\) 长度为 \(S+4\),也会有 \(1\) 到 \(a\) 长度为 \(S+6\) 的路径\(.......\)
那么我们可以知道:若 \(1\) 到 \(a\) 有一条长度为 \(S\) 的路径,其中 \(S \leq L\) 且 \(S\equiv L\pmod{2}\),那么 \(1\) 到 \(a\) 有一条长度为 \(L\) 的路径。
为了统计更多的信息,我们要使得对于每个 \(a\) 求出的 \(S\) 最小,以及保证奇偶性一致,那么很明显就是分 奇\(/\)偶 求最短路了。
设 \(d[a][1/0]\) 表示 \(1\) 到 \(a\) 路径长度为 奇\(/\)偶 的最短路长度,特别的,若 \(1\) 到 \(a\) 没有路径长度为 奇\(/\)偶 的路径,则 \(d[a][1/0]\) 为正无穷。初始时 \(d[1][0]=0\) ,其他均为正无穷。
考虑到 "奇路径\(+1\)为偶路径" 与 "偶路径\(+1\)为奇路径" ,那么对于一条边 \((u,v)\) ,我们都可以尝试用 \(d[u][0]+1\) 去更新 \(d[v][1]\),尝试用 \(d[u][1]+1\) 更新 \(d[v][0]\)。
这里拿 \(SPFA\) 做例子(别问为什么是这个死了的算法,问就是边权为 \(1\) 卡不掉,\(dijkstra\)同),具体的,在 \(SPFA\) 的过程中,在队列里多维护一个信息,表示松弛该节点时是奇最短路还是偶最短路(记为 \(type \in\{ 0,1\}\) ),对于每次松弛的节点 \(u\) ,扫描 \(u\) 的每一条出边 \((u,v)\) ,尝试用 \(d[u][type]+1\) 更新 \(d[v][type \ xor \ 1]\),若更新成功且二元组 \((v,type \ xor \ 1)\) 不在队中,再把\((v,type \ xor \ 1)\)插入队尾,直至队列为空,这样就处理好了 \(d\) 数组。
对于每组询问\((a,L)\), \(Θ(1)\) 比较 \(d[a][L\&1]\) 与 \(L\) 的大小关系即可。
需要注意的是,可能会有 \(1\) 号节点与所有点都没有连边的毒瘤数据,此时根据题意,当 \(a=1\) 时,无论 \(L\) 的取值,答案都是 \(NO\) ,但初始化时 \(d[1][0]=0\) ,如果 \(L\equiv 0\pmod{2}\),则会输出 \(YES\)。此处需要特判一下,判断 \(1\) 号节点与所有点都没有连边的情况只需判断 \(1\) 号节点的表头是否为 \(0\) 即可(邻接表存边)
Code 部分
\(SPFA\)版本:
#include<cstdio>
#include<cstring>
#include<queue>
#define RI register int
using namespace std;
inline int read()
{
int x=0,f=1;char s=getchar();
while(s<'0'||s>'9'){if(s=='-')f=-f;s=getchar();}
while(s>='0'&&s<='9'){x=x*10+s-'0';s=getchar();}
return x*f;
}
const int N=100100,M=200100;
int n,m,Q;
int tot,head[N],ver[M],edge[M],Next[M];
void add(int u,int v,int w)//邻接表
{
ver[++tot]=v; edge[tot]=w; Next[tot]=head[u]; head[u]=tot;
}
struct data{//队中节点数据
int type;// 奇/偶 最短路
int name;//编号
};
int d[N][2];
int vis[N][2];
void SPFA()
{
memset(d,0x3f,sizeof(d));
d[1][0]=0;
queue<data>q;
q.push((data){0,1});
while(q.size())
{
data u=q.front();q.pop();vis[u.name][u.type]=0;//取出队头
for(RI i=head[u.name];i;i=Next[i])
{
int v=ver[i],w=edge[i];
if(d[u.name][u.type]+w<d[v][u.type^1])//能够更新
{
d[v][u.type^1]=d[u.name][u.type]+w;//更新
if(!vis[v][u.type^1])
{
vis[v][u.type^1]=1;//标记
q.push((data){u.type^1,v});//入队
}
}
}
}
}
int main()
{
n=read(),m=read(),Q=read();
for(RI i=1;i<=m;i++)
{
int u=read(),v=read();
add(u,v,1),add(v,u,1);//加边
}
SPFA();
while(Q--)
{
int a=read(),L=read();
if(d[a][L&1]<=L&&(a!=1||head[a])/*特判*/)
puts("Yes");
else
puts("No");
}
return 0;
}
\(dijkstra\)版本:
#include<cstdio>
#include<cstring>
#include<algorithm>
#define RI register int
using namespace std;
inline int read()
{
int x=0,f=1;char s=getchar();
while(s<'0'||s>'9'){if(s=='-')f=-f;s=getchar();}
while(s>='0'&&s<='9'){x=x*10+s-'0';s=getchar();}
return x*f;
}
const int N=100100,M=200100;
int n,m,Q;
int tot,head[N],ver[M],edge[M],Next[M];
void add(int u,int v,int w)//邻接表
{
ver[++tot]=v; edge[tot]=w; Next[tot]=head[u]; head[u]=tot;
}
int len;
struct data{//堆中节点数据
int tpye;// 奇/偶 最短路
int dis;//长度
int name;//编号
}heap[N];
void put(data x)//入堆操作
{
heap[++len]=x;
int now=len;
while(now)
{
int nxt=now/2;
if(heap[nxt].dis<=heap[now].dis)break;
swap(heap[nxt],heap[now]);
now=nxt;
}
}
void change()//堆顶出堆
{
heap[1]=heap[len--];
int now=1;
while(now*2<=len)
{
int nxt=now*2;
if(nxt+1<=len&&heap[nxt].dis>heap[nxt+1].dis)nxt++;
if(heap[now].dis<=heap[nxt].dis)break;
swap(heap[now],heap[nxt]);
now=nxt;
}
}
int d[N][2];
int vis[N][2];
void dijkstra()
{
memset(d,0x3f,sizeof(d));
d[1][0]=0;
put((data){0,0,1});
while(len)
{
data u=heap[1];change();//取出堆顶
if(vis[u.name][u.tpye])continue;
vis[u.name][u.tpye]=1;//标记
for(RI i=head[u.name];i;i=Next[i])
{
int v=ver[i],w=edge[i];
if(d[u.name][u.tpye]+w<d[v][u.tpye^1])//能够更新
{
d[v][u.tpye^1]=d[u.name][u.tpye]+w;//更新
put((data){u.tpye^1,d[v][u.tpye^1],v});//入堆
}
}
}
}
int main()
{
n=read(),m=read(),Q=read();
for(RI i=1;i<=m;i++)
{
int u=read(),v=read();
add(u,v,1),add(v,u,1);//加边
}
dijkstra();
while(Q--)
{
int a=read(),L=read();
if(d[a][L&1]<=L&&(a!=1||head[a])/*特判*/)
puts("Yes");
else
puts("No");
}
return 0;
}
Thanks for watching
题解 CSP2019-J2T4【加工零件】的更多相关文章
- 题解 P5663 【加工零件【民间数据】】
博客园体验更佳 讲讲我的做法 确定做法 首先,看到这道题,我直接想到的是递归,于是复杂度就上天了,考虑最短路. 如何用最短路 首先,看一张图 我们该如何解决问题? 问题:\(3\)做\(5\)阶段的零 ...
- P5663 加工零件
P5663 加工零件 题解 暴力搜索 搜索显然会TLE #include<iostream> #include<cstdio> #include<cstdlib> ...
- P5663 加工零件 题解
原题链接 简要题意: 给定一个图,每次询问从 \(x\) 节点开始,\(y\) 步能不能达到 \(1\) 号节点. 算法一 这也是我本人考场算法.就是 深搜 . 因为你会发现,如果 \(x\) 用 \ ...
- 洛谷 P5663 加工零件
题目传送门 解题思路: 最暴力的做法: bfs模拟,每次将一个阶段的所有点拿出来,将其所有直连的点都放进队列,知道本阶段结束,最后看1号点会不会在最后一个阶段被放入队列.(洛谷数据40分) 优化了一下 ...
- 2019CSP-J T4 加工零件
题目描述 凯凯的工厂正在有条不紊地生产一种神奇的零件,神奇的零件的生产过程自然也很神奇.工厂里有 n 位工人,工人们从 1 ∼n 编号.某些工人之间存在双向的零件传送带.保证每两名工人之间最多只存在一 ...
- 洛谷 P5663 加工零件 & [NOIP2019普及组] (奇偶最短路)
传送门 解题思路 很容易想到用最短路来解决这一道问题(题解法),因为两个点之间可以互相无限走,所以如果到某个点的最短路是x,那么x+2,x+4也一定能够达到. 但是如何保证这是正确的呢?比如说到某个点 ...
- 题解 P1248 【加工生产调度】
题目 某工厂收到了 n 个产品的订单,这 n 个产品分别在 A.B 两个车间加工,并且必须先在 A 车间加工后才可以到 B 车间加工. 某个产品 i 在 A.B 两车间加工的时间分别为 Ai,Bi 怎 ...
- [题解]CSP2019 Solution - Part B
\(\text{orz}\) 一波现场 \(\text{A}\) 掉 \(\text{D1T3}\) 的神仙 D2T3 centroid Solution 考虑每个点 \(u\) 作为重心的贡献 假设 ...
- [题解]CSP2019 Solution - Part A
至于为什么是 \(\text{Part A}\) 而不是 \(\text{Day 1}\) 那是因为 Day1 T3 还没改 (那这六题的 \(\text{solution}\) 就按难度顺序写吧) ...
随机推荐
- Python学到什么程度可以面试工作(解答一)
本文整理了 26 个 Python 有用的技巧,将按照首字母从 A~Z 的顺序分享其中一些内容. all 或 any 人们经常开玩笑说 Python 是“可执行的伪代码”,但是当你可以这样编写代码时, ...
- Essential C++学习笔记
1.当我们调用一个函数时,会在内存中建立起一块特殊区域,称为“程序栈”,这块特殊区域提供了每个函数参数的存储空间,它也提供函数所定义的每个对象的内存空间--我们将这些对象称为局部对象.一旦函数完成,这 ...
- Spring 加定时器
定时器功能我们一般不常用, 但是一旦用到,那也是非常重要的, 今天我们就讲一下如何简单快速的使用定时器 第一种方法, 使用注解的方式完成定时器 1.在spring-servlet.xml文件中加入ta ...
- 基于bootstrap的下拉选择 ( combox ) 输入 ( input ) 功能
需求: 在编辑数据的时候,既可以让用户输入,也可以从下拉框中选择 思路: 参照下面的效果图,因为是表格里面的数据,所以下拉框触发按钮和输入框分别为1列,输入列可以设置是否输入(方法:<td co ...
- SharePoint REST 上传文件请求403错误
最近,需要在SharePoint上传文件到文档库,但是,上传的过程报错了. 错误代码 { "error": { "code": "-213057525 ...
- Educational Codeforces Round 80 (Rated for Div. 2)
A. Deadline 题目链接:https://codeforces.com/contest/1288/problem/A 题意: 给你一个 N 和 D,问是否存在一个 X , 使得 $x+\lce ...
- C# 调用word进程操作文档关闭进程
C# 调用word进程操作文档关闭进程 作者:Jesai 时间:2018-02-12 20:36:23 前言: office办公软件作为现在主流的一款办公软件,在我们的日常生活和日常工作里面几乎每天都 ...
- restframewor 版本(version)
1.路由 a.一级路由 from django.contrib import admin from django.urls import path, include from api import u ...
- python常用内置模块-random模块
random模块:用于生成随机数 '''关于数据类型序列相关,参照https://www.cnblogs.com/yyds/p/6123692.html''' random() 随机获取0 到1 之间 ...
- 「 优质资源20190409 」Java最新精选优质资源!
资源导读 经过小编精心整理,java最新优质资源出炉 不想看书,可以看视频,比较生动有趣,好的视频教程是一个好老师! 资源来自于网络,请勿用于商业用途 资源目录 1.Java Spring 技术栈构建 ...