【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;
}

  

posted @ 2018-09-23 10:16  cheng-qing  阅读(254)  评论(0编辑  收藏  举报