码迷,mamicode.com
首页 > 其他好文 > 详细

Gym - 101196:F Removal Game(区间DP)

时间:2018-08-04 21:41:20      阅读:180      评论:0      收藏:0      [点我收藏+]

标签:div   \n   最小   nbsp   name   std   int   ==   can   

题意:一个环状数组,给定可以删去一个数,代价的相邻两个数的gcd,求最小代价。

思路:区间DP即可,dp[i][j]表示[i,j]区间只剩下i和j时的最小代价,那么dp[i][j]=min  dp[i][k]+dp[k][j]+gcd(a[[i],a[j])。带上注意不能加倍做,以为常数会乘8,TLE。这也是这道题通过率低的原因。 

#include<bits/stdc++.h>
using namespace std;
const int maxn=110;
const int inf=1e9+7;
int dp[maxn][maxn],a[maxn];
int main()
{
    int N,i,j,k,ans;
    while(~scanf("%d",&N)&&N){
        for(i=1;i<=N;i++) for(j=1;j<=N;j++) dp[i][j]=inf;
        for(i=1;i<=N;i++) scanf("%d",&a[i]);
        for(j=1;j<N;j++){
          for(i=1;i<=N;i++){
                  if(j==1){
                     if(i+j==N) dp[i][N]=0;
                   else dp[i][(i+j)%N]=0; 
                }
                else for(k=i+1;k<i+j;k++){
                    int tj=(i+j)%N; if(!tj) tj=N;
                    int tk=k%N; if(!tk) tk=N;
                    dp[i][tj]=min(dp[i][tj],dp[i][tk]+dp[tk][tj]+__gcd(a[i],a[tj]));
                }
            }
        }
        ans=dp[1][N]+__gcd(a[1],a[N]);
        for(i=1;i<=N;i++)
         for(j=i+1;j<=N;j++)
          ans=min(ans,dp[i][j]+dp[j][i]+__gcd(a[i],a[j]));
        printf("%d\n",ans);
    }
    return 0;
}

 

Gym - 101196:F Removal Game(区间DP)

标签:div   \n   最小   nbsp   name   std   int   ==   can   

原文地址:https://www.cnblogs.com/hua-dong/p/9419904.html

(0)
(0)
   
举报
评论 一句话评论(0
登录后才能评论!
© 2014 mamicode.com 版权所有  联系我们:gaon5@hotmail.com
迷上了代码!