题目描述
输入
输出
样例输入
4 1
1 1 1 1
1 9 9 1
9 9 1 1
1 9 9 1
样例输出
23
题解
只可意会不可言传的非常神的背包dp,非常难写题解因此写的不好不要怪我。。。
考虑一个车站是怎么经过的,共有以下四种情况:
从后面来、往后面走,代价为d+v;
从前面来、往前面走,代价为u+e;
从后面来、往前面走,代价为d+e;
从前面来、往后面走,代价为u+v。
那么先不考虑从0直接到n+1的必经行程,中间的部分一定是一个回路。
从左向右考虑回路的每一段,必然是若干个向右的路线。把向右的路线看作左括号,向左的看作右括号,那么有:
第1种情况为‘((‘,第2种情况为‘))‘,第3种情况为‘)(‘,第4种情况为‘()‘。答案就是合法的括号序列。
设 $f[i][j]$ 表示前 $i$ 个位置,多余的‘(‘数目为 $2j$ 的最小代价。那么考虑这4种情况即可。注意第3、4种情况可能出现多次,因此是完全背包。
转移之间需要加上回路的代价 $T·2j$ 。
最后加上从0到n+1的代价 $(n+1)T$ 即为答案。
时间复杂度 $O(n^2)$
#include <cstdio> #include <cstring> #include <algorithm> using namespace std; int f[3010][3010]; int main() { int n , m , i , j , u , v , d , e; scanf("%d%d" , &n , &m); memset(f , 0x3f , sizeof(f)) , f[0][0] = 0; for(i = 1 ; i <= n ; i ++ ) { scanf("%d%d%d%d" , &u , &v , &d , &e); for(j = 1 ; j <= n ; j ++ ) f[i - 1][j] += j * m * 2; for(j = 1 ; j <= n ; j ++ ) f[i][j] = min(f[i][j] , f[i - 1][j - 1] + d + v); for(j = 0 ; j < n ; j ++ ) f[i][j] = min(f[i][j] , f[i - 1][j + 1] + u + e); for(j = 1 ; j <= n ; j ++ ) f[i][j] = min(f[i][j] , f[i - 1][j] + d + e); for(j = 0 ; j <= n ; j ++ ) f[i][j] = min(f[i][j] , f[i - 1][j] + u + v); for(j = 1 ; j <= n ; j ++ ) f[i][j] = min(f[i][j] , f[i][j - 1] + d + v); for(j = n - 1 ; ~j ; j -- ) f[i][j] = min(f[i][j] , f[i][j + 1] + u + e); } printf("%d\n" , f[n][0] + (n + 1) * m); return 0; }