HDU 5900(区间DP)

HDU 5900 QSC and Master

题意:给一串数的key和value,如果相邻两元素key不是互质的就可以将这俩移除并获得这俩的value值,移除后两侧的元素便是相邻了,问最终最大能获得多少value值。

思路:区间DP,区间长度为1时dp[i][i]为0,添加一个标记数组vis[i][j],记录区间内是否所有数字都为可以移除,每次处理需要讨论最后留下区间两侧的情况。

#include <cstdio>
#include <cstring>
#include <iostream>
#include <algorithm>
using namespace std;
typedef long long LL;
int key[505],val[505],n;
bool vis[505][505];//1表示区间内全部用到,0表示区间内有没用到的。
LL dp[505][505];
int gcd(int x,int y){
    return y==0?x:gcd(y,x%y);
}
int main(){
    int t; cin>>t;
    while(t--){
        cin>>n;
        for(int i=1;i<=n;i++)
            cin>>key[i];
        for(int i=1;i<=n;i++)
            cin>>val[i];
        memset(dp,0,sizeof(dp));
        memset(vis,0,sizeof(vis));
        for(int i=1;i<n;i++){
            if(gcd(key[i],key[i+1])!=1){
                dp[i][i+1]=val[i]+val[i+1];
                vis[i][i+1]=1;
            }
        }
        for(int l=3;l<=n;l++)
        for(int i=1;i<=n-l+1;i++){
            int j=i+l-1;
            for(int k=i;k<j;k++){
                if(vis[i][k]&&vis[k+1][j])
                    vis[i][j]=1;
                dp[i][j]=max(dp[i][j],dp[i][k]+dp[k+1][j]);
            }
            if(vis[i+1][j-1]&&gcd(key[i],key[j])!=1){
                vis[i][j]=1;
                dp[i][j]=dp[i+1][j-1]+val[i]+val[j];
            }
        }
        cout<<dp[1][n]<<endl;
    }

    return 0;
}

Psong

时间: 2024-10-11 06:31:23

HDU 5900(区间DP)的相关文章

2016 ACM/ICPC Asia Regional Shenyang Online 1009/HDU 5900 区间dp

QSC and Master Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 131072/131072 K (Java/Others)Total Submission(s): 859    Accepted Submission(s): 325 Problem Description Every school has some legends, Northeastern University is the same. Enter

hdu 5396 区间dp+组合

http://acm.hdu.edu.cn/showproblem.php?pid=5396 Problem Description Teacher Mai has n numbers a1,a2,?,anand n?1 operators("+", "-" or "*")op1,op2,?,opn?1, which are arranged in the form a1 op1 a2 op2 a3 ? an. He wants to erase

HDU 5115 区间DP

有n只狼,每只狼有两种属性,一种攻击力一种附加值,每杀一只狼 受到的伤害值为这只狼的攻击值与它旁边的两只狼的附加值的和,求把所有狼都杀光受到的最小的伤害值. 注意:如果杀死中间的狼,两边的狼会紧凑过来,也就是说中间不存在空位 很明显的区间DP dp[i][k]=Min(dp[i][k],dp[i][l-1]+dp[l+1][k]+a[l]+b[i-1]+b[k+1]); // i位置起始到k位置,最后杀死k位置的狼 #include "stdio.h" #include "s

hdu 4283 区间dp

You Are the One Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others)Total Submission(s): 3878    Accepted Submission(s): 1793 Problem Description The TV shows such as You Are the One has been very popular. In order to m

HDU 5273 区间DP

输入一组数,m次询问 问每一个询问区间的逆序数有多少 区间DP简单题 #include "stdio.h" #include "string.h" int dp[1010][1010],a[1010]; int main() { int n,m,i,j,k; while (scanf("%d%d",&n,&m)!=EOF) { for (i=1;i<=n;i++) scanf("%d",&a[i]

HDU 5396 区间DP 数学 Expression

题意:有n个数字,n-1个运算符,每个运算符的顺序可以任意,因此一共有 (n - 1)! 种运算顺序,得到 (n - 1)! 个运算结果,然后求这些运算结果之和 MOD 1e9+7. 分析: 类比最优矩阵链乘,枚举区间[l, r]中最后一个运算符的位置k. 如果运算符为乘法的话,那么根据乘法分配率这个乘法会分配进去. 这个区间中一共有r - l个运算符,其中最后一个运算符已经定了是第k个,左区间[l, k]有k - l个运算符,右区间[k + 1, r]有 r - k - 1 个运算符. 而且左

hdu 4293 区间DP

1 /* 2 题目大意:n个人分成若干组,每个人都描叙他们组前面有多少人后面有多少人, 3 求说真话的人最多有多少个. 4 解题思路:把同一组的人数统计起来他们组前面有x人后面有y人, 5 num[x+1][n-y]表示区间[x+1,n-y]的权值,num[x+1][n-y]<=n-x-y 6 那么就是求不重合,[1,n]区间的最大值 7 */ 8 #include <iostream> 9 #include <cstdio> 10 #include <cstring&

hdu 4745 区间dp

题意:求一个环的最长回文序列,是序列不是串 链接:点我 起点是可以任意的, 所以只要求出每个区间的最长回文序列之后取max(dp[1][i]+dp[i+1][n]),即可得最终答案 本来是想扩展两倍的,但是后来的最大不太好想 将 环倍增成链,求出窗口为n的最长子序列,但这不是最终的解,你可以试看看Sample 2,是只能得出4,因为它在选中的回文外面还可以选中一个当做起点来跳,所以外面得判断找出来的回文外面是否还有可以当起点的石头,即可以找窗口为(n- 1)的长度+1.所以解即找 窗口为n的长度

hdu 4632区间dp

给你一个字符串  让你求最大回文子序列的个数:dp[i][j]表示i到j的最大回文子序列个数(注意是个数,就要有累加过程  如果是最大长度  就直接赋值) 首先dp[i][j]=dp[i+1][j]+dp[i][j-1] 中间有重复的()就是与i+1余j-1无关的   所以还得减去dp[i+1][j-1] 然后判断str[i]与str[j]是否相等,如果是再加上dp[i+1][j-1]+1: #include<stdio.h> #include<string.h> #include

hdu 2476 区间dp

题意是让你把第一个字符串变为第二个的最小步数,具体变法就不说了, 假设第一个和第二个串对应位置全都不一样,那就先不考虑串1了,dp[i][j]i到j的最小步数:很显然开始dp[i][j]=dp[i+1][j] 我是从后往前跑的,如果中间处在相等比如str[i]=str[k] 则dp[i][k]=dp[i+1][k]  因为如果刷i到k-1要x步刷i+1到k要y步,现在str[i]==str[k]   如果第k个不和第i个一起刷  及先刷i再刷i+1到k-1 再刷k 则需要x+1步 其实如果i和k