【P1078】 文化之旅 bfs题解

 一道数据十分水的题目,出自2012年的普及组。各种玄学算法都能过,最常见的是倒着搜的dfs。

我是在水最短路的时候刷到这个题的,因为喜欢用bfs,所以写了这个题解。

思路来自:洛谷dalao   lushangyin的题解

这个题的数据很小,n只有100,直接使用二维数组保存数据  

 也很像前向星的存边方法

定义

        a[x][i].to表示点x的第i条边所到达的点

        a[x][i].dis表示这条边的权值

        mapp[i][j]表示文化 与文化 之间的关系

        c[i]表示点 i 所有的文化

        vis[i][j]判断点 i 与点 j 之间文化目前状态

        cnt[i] 表示点i 已经有几条边了

#include<bits/stdc++.h> //大家不要学
using namespace std;
struct gjc{//仅出自个人习惯
	int to;
	int dis;
} a[110][110];
int cnt[110];
int n,m,k,str,en;
int c[110],d[110];
int mapp[110][110];
int vis[2001][101];
void read(){//初始化
	cin>>n>>k>>m>>str>>en;
	for(int i=1;i<=n;i++)
	  cin>>c[i];
	for(int i=1;i<=k;i++)
	  for(int j=1;j<=k;j++)
	    cin>>mapp[i][j];
	for(int i=1;i<=m;i++)
	{
		int x,y,z;
		cin>>x>>y>>z;
		cnt[x]++;cnt[y]++;//边数加一
		a[x][cnt[x]].to=y;a[x][cnt[x]].dis=z;//希望你能理解
		a[y][cnt[y]].to=x;a[y][cnt[y]].dis=z;
	}
}
void out()//输出
{
	if(d[en]==0) cout<<"-1";
	else cout<<d[en];
}

void bfs(){//不用解释
	priority_queue<int,vector<int>,greater<int> > que;//定义一个小根堆优化的优先队列 后来
                                                      //发现普通的队列好像也可以
	int node;
	node=str;
	que.push(node);//将起点推入队列
	for(int i=1;i<=n;i++) 
	    if(mapp[c[i]][c[str]]==1) vis[str][i]=1;
	    else if(c[i]==c[str])vis[str][i]=1;//改变文化状态  根据题目大意很好理解
	while(!que.empty())
	{
		int v=que.top();que.pop();//取出队首元素(是一个点)
		for(int i=1;i<=cnt[v];i++)//遍历这个点的所有边
		{
		    node=a[v][i].to;//node存储下一个点的编号
			if(vis[v][node]!=0) continue;//如果当前点和目标点文化有冲突就跳过
			if(d[node]>a[v][i].dis+d[v]||d[node]==0)//如果这么走更短或是第一次走
			{
			    
				for(int j=1;j<=n;j++)//和上面对起点文化的改变很像
				{
				    if(mapp[c[j]][c[node]==1])	vis[node][j]=1;
				    else 
				      if(vis[v][j]==1) vis[node][j]=1;//理解一下
				    else 
					  if(c[j]==c[node]) vis[node][j]=1;
 				}
				d[node]=d[v]+a[v][i].dis;
				que.push(node);//将缩短路径的点推入队列中
			}
		}
	}
} 


int main(){
	read();
	bfs();
        out();
 	return 0;
}

 

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值