P5663 [CSP-J2019] 加工零件

因某wsx248老师强迫建议而写的 第一篇博客
有错误恳请指证
一道比较容易 想到的题,可我竟然想了好久还看了好久的题解 才稍微明白了。
首先可以得出ai点(至少?)会做L,L-2,L-4,L-6,L-8…等零件,旁边的点至少做L-1,L-3,L-5,L-7…等零件。
又因为将问题转化为是否存在一条从ai到1(是不是和其他的不太一样,可以看q2),长度为L的路径(可以走重复边,点),所以可以先思考如果保证L为偶数(或奇数)时怎样判定为Yes还是No。
可以做从ai到1的长度为偶数的最短路,以确保1号点有零件做。如果他有零件做,那么是不是可能会做到原材料,如果不可能就是1,3…级材料
当长度为奇数时,同理可得。
那么有几个问题
q1.如何求长度为奇数和偶数的最短路?

ans1:如果单独求奇数或偶数的最短路,不太好思考。但是两个都求出来,长度为奇数就可以由长度为偶数的转移过来,偶数就可以由长度为奇数的转移过来(因为边权全为一)。若还不太懂可以看看代码。

q2:每次都从ai到1做一遍bfs,时间复杂度不会爆吗?

ans2:由于求出最短路只会用到到1的值,又因为是双向边,就可以等价于从1到所有的点做bfs,后面有点ai用就行了,就达到了预处理O(n),查询O(1)。

q3:代码中不需要处理不连通的特殊情况吗?

ans3:之所以不用特殊处理不连通的情况,是因为若不连通,dis的值会是0x3f3f3f3f(无穷大)。
code:

#include <bits/stdc++.h>
using namespace std;
inline int read(){
	int w=0,f=1;
	char ch=getchar();
	while(ch<'0'||ch>'9'){
		if(ch=='-') f=-1;
		ch=getchar();
	}
	while('0'<=ch&&ch<='9'){
		w=w*10+ch-'0';
		ch=getchar();
	}
	return w*f;
}
const int N=1e5+15;
int n,m,q,dis[N][2];//dis[i][0]为从1到i的偶数最短路,dis[i][1]为从1到i的奇数最短路。
vector <int> e[N];
struct Point{
	int p,w,z;//p是当前点,w是从1到当前点的最小距离,z是奇偶性质 (当z==0时为偶,当 z==1是为奇) 
};
void bfs(){
	queue <Point> q;
	memset(dis,0x3f,sizeof(dis));
	q.push(Point{1,0,0});
	dis[1][0]=0;
	while(!q.empty()){
		Point now=q.front();
		q.pop();
		int len=e[now.p].size(),w=now.w;
		for(int i = 0 ; i<len ; i++){
			int to=e[now.p][i];
			if(dis[now.p][now.z]+1<dis[to][now.z^1]){//运用^,将0改成1,1改成0.
				dis[to][now.z^1]=dis[now.p][now.z]+1;
				q.push({to,dis[to][now.z^1],now.z^1});
			}
		}
	}
}//之所以不用特殊处理不连通的情况,是因为若不连通,dis的值会是0x3f3f3f3f(无穷大)
int main(){
	n=read(),m=read(),q=read();
	for(int i = 1 ; i<=m ; i++){
		int u=read(),v=read();
		e[u].push_back(v);
		e[v].push_back(u);
	}
	bfs();
	for(int i = 1 ; i<=q ; i++){
		int a=read(),L=read();
		if(L>=dis[a][L&1]){
			puts("Yes");
		}
		else{
			puts("No");
		}
	}
	return 0;
}

如果有其他问题,可以@或私信我,会尽量回答的。

  • 9
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 3
    评论
评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值