• 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;
    }
    

      

  • 相关阅读:
    IIS应用程序池标识(程序池账户)ApplicationPoolIdentify
    笔记二 sql 基础知识
    笔记三 索引讲解
    SqlParameter使用参数错误,提示请求超时或则查询速度过慢
    SVN 常见操作
    Excel 基本操作
    sql server row_number分页排序
    本地部署IIS
    sql中去掉字段的所有空格
    sql取逗号前后数据与批量修改某一字段某一值
  • 原文地址:https://www.cnblogs.com/td15980891505/p/4956240.html
Copyright © 2020-2023  润新知