该题的难点在于存在两个“子动态规划”并且需要用一个中间数组将两个“子动态规划”联系起来。
解题思路:
- 计算任意两个村子之间的距离,用动态规范显然降低了不少复杂度。(都可以单独出一个题目了~~(╯﹏╰)~~)
- 在任意两个村子之间建一所小学,并计算两个村子之间的所有村子到该小学的最短距离,求解所有情况。
- 计算在第1~i个村子中间建j所小学的最短距离,f[i][j]=1~k个村子建j-1所小学的最短距离与k+1~i个村子建一所小学(正好契合了中间数组)的最短距离之和,使用动态规划再合适不过。(又可以单独出一个题目了~~(╯﹏╰)~~)
1 #include<iostream> 2 using namespace std; 3 4 const int N=505,MAX_DIS=0x7fffffff; 5 int a[N][N],b[N][N],f[N][N]; 6 int main(){ 7 int m,n; 8 cin>>m>>n; 9 for(int i=1;i<m;i++){ 10 cin>>a[i][i+1]; 11 a[i+1][i]=a[i][i+1]; 12 } 13 //计算任意两个村子之间的距离 14 for(int i=1;i<m;i++){ 15 for(int j=i+2;j<=m;j++){ 16 a[i][j]=a[j][i]=a[i][j-1]+a[j-1][j]; 17 } 18 } 19 //在任意两个村子之间建一所学校,计算两村子之间的所有村子到该学校的最短距离 20 for(int i=1;i<m;i++){ 21 for(int j=i+1;j<=m;j++){ 22 int pos=(i+j)/2;//两村子中间建学校距离最短? 23 for(int k=i;k<=j;k++){ 24 b[i][j]+=a[k][pos]; 25 } 26 } 27 } 28 //f[i][j]表示在前i个村子(1~i)里建j所小学的最短距离 29 //初始化 30 for(int i=1;i<=m;i++){ 31 f[i][1]=b[1][i]; 32 } 33 for(int i=3;i<=m;i++){ 34 for(int j=2;j<=n;j++){ 35 f[i][j]=MAX_DIS; 36 for(int k=1;k<i;k++) 37 f[i][j]=min(f[i][j],f[k][j-1]+b[k+1][i]); 38 } 39 } 40 cout<<f[m][n]; 41 return 0; 42 }
做了有好几个动态规划了(当然这题是披着递推的动态规划)
颇有几分心得:
如果你要看懂别人的动态规划题解,那么你必须知道其中二维数组的含义,比如a[i][j],你需要知道i、j和a[i][j]分别表示什么。
如果你要会写动态规划,那就努力去找其中的递推关系,或者说是大输入与小输入之间的联系,然后再用(一般)二维数组表达它们。
但难点往往在于:
- 有的递推关系(联系)很难找出
- 即使得到了一点递推关系的头绪,却又没有用二维数组表达它们的能力(像这题,需要使用多个二维数组表达多种联系)