1.E - Endless Holidays--原题链接
2.题目截图






3.思路历程
题目明确说明了,晚上会进行选择,留在当前城市或者移动到下一个城市,想要每天都在假日城市里待着,这里说了,一次只能动一次(赛时理解错了,以为随便跑,我说怎么这么简单写了个并查集发现样例过不了直接急哭了)。然后理解这个,思路就很明确了。
我们每次只能从假期城市移动到假期城市,而且每周最多只有十天,数据很小,我们直接考虑分层图,把每一天的数据都存下来,然后只有两个均为假日且有联通在存为有通路,理解这个逻辑并构建出分层图之后,我们剩下的就判断是否成环就行,这里我是用的拓扑排序染色法写的,有环yes没环no,这题就a了
4.ac代码
cpp
#include<bits/stdc++.h>
#define int long long
#define endl "\n"
using namespace std;
const int N=100000;
array<int,2> ed[N*3+5];
string s[N+5];
vector <int> h[N*10+5];
int color[N*10+5];//三色颜色,对应三种状态,0未遍历,-1正在遍历,1遍历完成
bool tin = 1; // 是否多组测试,默认为单组
bool dfs(int x)//染色法拓扑排序
{
color[x]=-1;//正在遍历
for(auto y:h[x])
{
if(color[y]==-1)//遇到正在便利的,说明是环
return 1;
if(!color[y])//没遍历过的
{
if(dfs(y))//继续搜下去
return 1;
}
}
color[x]=1;//遍历结束
return 0;
}
bool have_cycle(int n,int w)//检查有无环
{
int e=n*w;//分层之后总的节点数
for(int i=1;i<=e;i++)//重置颜色数组
{
color[i]=0;
}
for(int i=1;i<=n;i++)
{
if(!color[i])//遇到没遍历的进行染色,因为规定了第一天去,所以只有1-n
{
if(dfs(i))
return 1;
}
}
return 0;
}
void solve() {
int n,m,x,y;
cin>>n>>m;
for(int i=0;i<m;i++)
{
cin>>x>>y;
ed[i*2]={x,y};//双向存单向,先存下来
ed[i*2+1]={y,x};//先
}
int w,ue=2*m+n;
cin>>w;
for(int i=1;i<=n;i++)
cin>>s[i];
for(int i=0;i<n;i++)
{
ed[i+2*m]={i+1,i+1};//因为可以留下来,所以存一个自己到自己
}
for(int i=1;i<=n;i++)
{
for(int j=0;j<w;j++)
{
h[n*j+i].clear();//重置图
}
}
for(int i=0;i<ue;i++)
{
for(int j=0;j<w;j++)
{
int neday=(j+1)%w;
if(s[ed[i][0]][j]=='o'&&s[ed[i][1]][neday]=='o')//可以从j的ed[i][0],到达下一天的ed[i][1],存图
{
h[n*j+ed[i][0]].push_back(n*neday+ed[i][1]);//这里n*j和n*neday是为了完成分层
}
}
}
if(have_cycle(n,w))//有环yes没环no
{
cout<<"Yes"<<endl;
}
else
cout<<"No"<<endl;
}
signed main() {
ios::sync_with_stdio(0); cin.tie(0), cout.tie(0);
int T = 1;
if (tin) cin >> T;
while (T--) {
solve();
}
return 0;
}
翻译用的deepseek,还挺不错的