• HDU1087上升子序列的最大和


       解法一

          此题是一个简单的动态规划问题,用dp[i]记做最后一步经过第i个数所得到的最大sum值,则结果=max(dp[i]),i=1,...n.考虑dp[i]的前一步会经过那里?假设dp[i]的前一步经过第j个数,则子问题dp[j]满足最优子结构。dp[i]=a[i]+max(dp[j]) .(a[j]<a[i]);

    /*---dp[i]表示最后一步经过第i个数
    ----转移方程dp[i]=a[i]+max(dp[j])(a[j]<a[i],j<i)
    */
    #define _CRT_SECURE_NO_DEPRECATE
    #include<iostream>
    #include<vector>
    #include<string.h>
    #include<algorithm>
    using namespace std;
    typedef long long LL;
    const int MAXN = 1000 + 10;
    LL dp[MAXN];
    int a[MAXN];
    int main(){
    	int n, i, j;
    	while (scanf("%d", &n) && n){
    		for (i = 1; i <=n; i++)
    			scanf("%d", &a[i]);
    		memset(dp,0, sizeof(dp));
    		dp[1] = a[1];
    		LL ans =dp[1];
    		for (i = 2; i <= n; i++){
    			dp[i] = a[i];
    			for (j = 1; j < i; j++){
    				if (a[j]<a[i])
    				  dp[i] = max(dp[i], dp[j]+a[i]);
    			}
    			ans = max(ans, dp[i]);
    		}
    		printf("%I64d
    ", ans);
    	}
    	return 0;
    }
    

      解法二  转化为GAD模型

            考虑若i<j,同时选择a[i]和a[j]的合法条件是a[i]<a[j].于是在数组中寻找满足i<j且a[i]<a[j],将i向j连接一条有向边。用dp[i]表示从i出发可以得到的最大和,则有:

    dp[i]=a[i]+max(dp[j]) .j是i的邻接点。采用记忆化搜索来求解。

      

    /*---DAG模型求解
    ----若序列中a[i]<a[j](i<j)则从i到j连接一条有向边
    */
    #define _CRT_SECURE_NO_DEPRECATE
    #include<iostream>
    #include<vector>
    #include<string.h>
    #include<algorithm>
    using namespace std;
    typedef long long LL;
    const int MAXN = 1000 + 10;
    LL dp[MAXN];
    int a[MAXN];
    vector<int>vec[MAXN];
    //记忆化搜索
    LL dfs(int i){
    	LL &ans = dp[i];
    	if (ans >= 0)
    		return ans;
    	ans = 0;
    	for (int j = 0; j < (int)vec[i].size(); j++){
    		ans = max(ans,dfs(vec[i][j]));
    	}
    	return ans=ans + a[i];
    }
    int main(){
    	int n, i,j;
    	while (scanf("%d", &n) && n){
    		for (i = 0; i < n; i++)
    			scanf("%d", &a[i]);
    		for (i = 0; i < n; i++){
    			vec[i].clear();
    			for (j = i + 1; j < n; j++){
    				if (a[j]>a[i])
    					vec[i].push_back(j);  //i到j有向边
    			}
    		}
    		LL ans = 0;
    		memset(dp, -1, sizeof(dp));
    		for (i = 0; i < n; i++){
    			if (dp[i]<0)
    				dp[i] = dfs(i);
    			ans = max(ans, dp[i]);
    		}
    		printf("%I64d
    ", ans);
    	}
    	return 0;
    }
    

      

  • 相关阅读:
    oracle-sql脚本
    vue生命周期
    使用vue搭建项目(创建手脚架)
    bootcss
    miniMobile(手机)
    mui(手机)
    layui
    Element
    如何学好Spring
    使用Vue做评论+localStorage存储(js模块化)
  • 原文地址:https://www.cnblogs.com/td15980891505/p/4956240.html
Copyright © 2020-2023  润新知