题意:就是给你一个图 让你从第一列开始走到最后一列 寻找最小的路径,然后这一题难点在于 路径的记录以及第一行跟最后一行联通 最后一行跟第一行也是联通的
题目传送门
代码如下,紫书讲的很好了我就不写分析了,代码又解释
#include <bits/stdc++.h>
using namespace std;
int a[15][105],d[15][105];// a数组存图,d数组进行dp模拟,选择最优路径
int Next[15][105];//存路径 每一个next数组记录的是下一个路径,第一行的路径是first记录的
const int INF=0x3f3f3f;//最大值
int main()
{
int m,n;
while(cin>>m>>n)
{
for(int i=0;i<m;i++)
{
for(int j=0;j<n;j++)
{
cin>>a[i][j];//存图
}
}
int ans=INF,first=0;//first记录第一行的路径
for(int j=n-1;j>=0;j--)//逆推 容易记录路径
{
for(int i=0;i<m;i++)
{
if(j==n-1)
{
d[i][j]=a[i][j];
}
else
{
int row[3]={i,i-1,i+1};//三个方向的dp
if(i==0) row[1]=m-1;//边界问题 这个图是一个环 第一行跟最后一行连同,同理最后一行跟第一行也是联通
if(i==m-1) row[2]=0;
sort(row,row+3);
d[i][j]=INF;
for(int k=0;k<3;k++)
{
int v=d[row[k]][j+1]+a[i][j];
if(v<d[i][j])
{
d[i][j]=v;
Next[i][j]=row[k];
}
}
}
if(j==0&&ans>d[i][j])
{
ans=d[i][j];//记录最大值
first=i;
}
}
}
cout<<first+1;
for(int i=Next[first][0],j=1;j<n;i=Next[i][j],j++)
{
cout<<" "<<i+1;//输出路径
}
cout<<endl<<ans<<endl;//输出答案
}
return 0;
}