有一个 \(n\times m\) 的网格,方格上有 \(k\) 个宝藏,一个人从 \((1,1)\) 出发,可以向左或者向右走,但不能向下走。给出 \(q\) 个列,在这些列上可以向上走,其他列不能向上走。可以重复经过同一个点。求从 \((1,1)\) 出发,经过所有宝藏的最短路径长度。\(n,m,k,q\leq 2\times 10^5\)
Solution
设 \(f[i][0/1]\) 表示从 \((0,0)\) 走到 \(i\) 行的左侧/右侧,并遍历了 \(1 \to i-1\) 行的所有宝物的最短路径长度
(挺怕这种题的)
#include
using namespace std;
#define int long long
const int N = 500005;
int n,m,k,q,l[N],r[N],f[N][2],b[N],pl[N],pr[N];
int dist(int p,int q) {
if(p>q) swap(p,q);
if(pr[p]<=q || pl[q]>=p) return 0;
return min(pr[q]-q,p-pl[p]);
}
signed main() {
ios::sync_with_stdio(false);
cin>>n>>m>>k>>q;
for(int i=1;i<=n;i++) l[i]=1e9, r[i]=0;
for(int i=1;i<=k;i++) {
int t1,t2;
cin>>t1>>t2;
l[t1]=min(l[t1],t2);
r[t1]=max(r[t1],t2);
}
for(int i=1;i<=q;i++) {
int t1;
cin>>t1;
b[t1]=1;
}
pl[0]=-1e9;
for(int i=1;i<=m;i++) {
if(b[i]==1) pl[i]=i;
else pl[i]=pl[i-1];
}
pr[m+1]=1e9;
for(int i=m;i>=1;--i) {
if(b[i]==1) pr[i]=i;
else pr[i]=pr[i+1];
}
if(r[1]) f[1][0]=l[1]-1, f[1][1]=r[1]-1;
else l[1]=1,r[1]=1,f[1][0]=f[1][1]=0;
int ans=f[1][1];
for(int i=2;i<=n;i++) {
if(r[i]==0) {
f[i][0]=f[i-1][0];
f[i][1]=f[i-1][1];
l[i]=l[i-1];
r[i]=r[i-1];
++f[i][0]; ++f[i][1];
continue;
}
f[i][0]=min(f[i-1][0]+2*dist(r[i-1],l[i])+abs(l[i]-r[i-1])+r[i-1]-l[i-1],
f[i-1][1]+2*dist(l[i-1],l[i])+abs(l[i]-l[i-1])+r[i-1]-l[i-1]);
f[i][1]=min(f[i-1][0]+2*dist(r[i-1],r[i])+abs(r[i]-r[i-1])+r[i-1]-l[i-1],
f[i-1][1]+2*dist(l[i-1],r[i])+abs(r[i]-l[i-1])+r[i-1]-l[i-1]);
++f[i][0]; ++f[i][1];
//cout<<"i="<