动态规划-选学生
2016-09-18 12:57
190 查看
题目
有 n 个学生站成一排,每个学生有一个能力值,从这 n 个学生中按照顺序选取 k 名学生,
要求相邻两个学生的位置编号的差不超过 d,
使得这 k 个学生的能力值的乘积最大,你能返回最大的乘积吗
输入描述:
每个输入包含 1 个测试用例。
每个测试数据的第一行包含一个整数 n (1 <= n <= 50),
表示学生的个数,
接下来的一行,包含 n 个整数,
按顺序表示每个学生的能力值 ai(-50 <= ai <= 50)。
接下来的一行包含两个整数,
k 和 d (1 <= k <= 10, 1 <= d <= 50)。
输出描述:
输出一行表示最大的乘积。
思路
因为有正有负,所以需要两个数组,最大最小,Fmax,Fmin K*N维的,K是选几人,N是总共所少人
读数据,动态分配释放内存
Fmax[k][i]表示以i个人结束,肯定要选,然后已选了k个,下标从0开始和对应人数的要-1
k=0时,表示只选1个人,那么就是
i: 0~n-1总选最大的,初始化过程
k: 1~K-1, 与之前的状态k-1有关,然后最大间隔是D, 所以间隔
m: 1~D,学生位数就是
i-m,当
i-m>=0时
Fmax[k][i] = max(Fmax[k][i], max(Fmax[k-1][i-m]*ai[i],Fmin[k-1][i-m]*ai[i]));同理,
Fmin同样计算,但是是取小的
不断更新,k++
最后就从
Fmax[K-1][0~N]之间选择个最大的,就是结果
代码
/* 有 n 个学生站成一排,每个学生有一个能力值, 牛牛想从这 n 个学生中按照顺序选取 k 名学生, 要求相邻两个学生的位置编号的差不超过 d, 使得这 k 个学生的能力值的乘积最大,你能返回最大的乘积吗 输入描述: 每个输入包含 1 个测试用例。 每个测试数据的第一行包含一个整数 n (1 <= n <= 50), 表示学生的个数, 接下来的一行,包含 n 个整数, 按顺序表示每个学生的能力值 ai(-50 <= ai <= 50)。 接下来的一行包含两个整数, k 和 d (1 <= k <= 10, 1 <= d <= 50)。 输出描述: 输出一行表示最大的乘积。 */ #include<iostream> using namespace std; void display(long long ** a, int x, int y) { for(int i = 0;i<x;i++) { for(int j = 0;j<y;j++) cout<<a[i][j]<< " "; cout<<endl; } } // n 个学生中按照顺序选取 k 名学生, // 要求相邻两个学生的位置编号的差不超过 d long long getMaxAI(int *ai, int K, int D,int N) { long long maxAI = 0; long long **Fmax = new long long *[K]; long long **Fmin = new long long *[K]; for (int x =0;x<K;x++) { Fmax[x] = new long long (); //初始化为0;加上() Fmin[x] = new long long (); // 初始化,选一个人的时候,K=0,那么没得选,自己 if(x==0) { for (int y = 0;y<N;y++) { Fmax[x][y]=ai[y]; Fmin[x][y]=ai[y]; } } } //display(Fmax,K,N); for(int k = 1;k<K;k++) { for(int i = 0;i<N;i++ ) { for (int m =1;m<=D ; m++ ) { if(i-m>=0) { Fmax[k][i] = max(Fmax[k][i], max(Fmax[k-1][i-m]*ai[i],Fmin[k-1][i-m]*ai[i])); Fmin[k][i] = min(Fmin[k][i], min(Fmax[k-1][i-m]*ai[i],Fmin[k-1][i-m]*ai[i])); } } } //cout<<maxAI<<endl; } for (int n = 0;n<N;n++) { maxAI = max(maxAI,Fmax[K-1] ); } for (int x =0;x<K;x++) { delete []Fmax[x] ; delete []Fmin[x] ; } delete []Fmax; delete []Fmin; return maxAI; } int main() { int n; while(cin>>n) { int * ai = new int ; for(int i = 0;i<n;i++) { cin>>ai[i]; } int k,d; cin>>k>>d; cout<<getMaxAI(ai,k,d,n)<<endl; delete []ai; } //int ai2[3] = {7,4,7}; //int k2 =2 ,d2 =50, n2=3; //getMaxAI(ai2,k2,d2,n2); return 0; }
相关文章推荐
- 说说动态规划之矩阵链乘法
- 动态规划——357. Count Numbers with Unique Digits[medium]
- poj 3356 AGTC(动态规划:最短编辑距离)
- 【DP动态规划】个人常用基础动态规划DP小总结【TODO】
- 算法之动态规划总结
- 职业、人生的动态规划?
- 动态规划 (一) 基本概念
- 动态规划——数塔(hdu2084)
- 动态规划简单例子--象棋步骤
- 【练习09】简单动态规划 1004 命运
- 动态规划 ------- 最长公共子序列
- 动态规划之装配线调度理解
- JZOJ3418. 【NOIP动态规划专题】选课(2017.8DP&贪心专题)
- 动态规划之线性动规(钢条切割、合唱队、最长递增子序列)
- 动态规划——数字三角形问题(空间优化)
- 动态规划 UVa 624 CD
- 动态规划之任意顶点间最短距离
- 动态规划之编辑距离问题
- 动态规划 数塔问题
- 动态规划之0-1背包问题