Vijos1006P1006晴天小猪历险记之Hill[最短路]

时间:2021-10-05 13:22:48
P1006晴天小猪历险记之Hill

背景

在很久很久以前,有一个动物村庄,那里是猪的乐园(^_^),村民们勤劳、勇敢、善良、团结……
不过有一天,最小的小小猪生病了,而这种病是极其罕见的,因此大家都没有储存这种药物。所以晴天小猪自告奋勇,要去采取这种药草。于是,晴天小猪的传奇故事便由此展开……

描述

这一天,他来到了一座深山的山脚下,因为只有这座深山中的一位隐者才知道这种药草的所在。但是上山的路错综复杂,由于小小猪的病情,晴天小猪想找一条需时最少的路到达山顶,但现在它一头雾水,所以向你求助。

山用一个三角形表示,从山顶依次向下有1段、2段、3段等山路,每一段用一个数字T(1<=T<=100)表示,代表晴天小猪在这一段山路上需要爬的时间,每一次它都可以朝左、右、左上、右上四个方向走。山是环形的。(注意:在任意一层的第一段也可以走到本层的最后一段或上一层的最后一段)。

晴天小猪从山的左下角出发,目的地为山顶,即隐者的小屋。

格式

输入格式

第一行有一个数n(2<=n<=1000),表示山的高度。

从第二行至第n+1行,第i+1行有i个数,每个数表示晴天小猪在这一段山路上需要爬的时间。

输出格式

一个数,即晴天小猪所需要的最短时间。

样例1

样例输入1[复制]

5
1
2 3
4 5 6
10 1 7 8
1 1 4 5 6

样例输出1[复制]

10

限制

各个测试点1s

提示

在山的两侧的走法略有特殊,请自己模拟一下,开始我自己都弄错了……

来源

Sunnypig


一开始想DP

然后就.......了不就是个最短路注意一下建图

从上往下建比较好吧

两侧

if(i<n&&i!=1){
add(id(i,i),id(i+1,1));
add(id(i,1),id(i+1,i+1));
}
#include <iostream>
#include <cstdio>
#include <algorithm>
#include <cstring>
#include <cmath>
#include <queue>
using namespace std;
typedef long long ll;
const int N=*/,INF=1e9+;
inline int read(){
char c=getchar();int x=,f=;
while(c<''||c>''){if(c=='-')f=-;c=getchar();}
while(c>=''&&c<=''){x=x*+c-'';c=getchar();}
return x*f;
}
int n,t[N],u,v,w;
inline int id(int i,int j){
return i*(i-)/+j;
}
struct edge{
int v,ne;
}e[N<<];
int cnt=,h[N];
inline void add(int u,int v){
cnt++;
e[cnt].v=v;e[cnt].ne=h[u];h[u]=cnt;
}
inline void ins(int u,int v){
add(u,v);add(v,u);
}
int d[N],inq[N];
queue<int> q;
void spfa(int s){
int m=id(n,n);
for(int i=;i<=m;i++) d[i]=INF;
d[s]=;
q.push(s);
while(!q.empty()){
int u=q.front();q.pop();
inq[u]=;
for(int i=h[u];i;i=e[i].ne){
int v=e[i].v,w=t[u];
if(d[v]>d[u]+w){
d[v]=d[u]+w;
if(!inq[v]){inq[v]=;q.push(v);}
}
}
}
}
int main(){
n=read();
for(int i=;i<=n;i++){
if(i!=) ins(id(i,),id(i,i));
if(i<n&&i!=){
add(id(i,i),id(i+,));
add(id(i,),id(i+,i+));
}
for(int j=;j<=i;j++){
t[id(i,j)]=read();
if(j<i) ins(id(i,j),id(i,j+));
if(i<n) add(id(i,j),id(i+,j)),add(id(i,j),id(i+,j+));
}
}
spfa(id(,));
printf("%d",d[id(n,)]+t[id(n,)]);
}