每个输入包含 1 个测试用例。每个测试数据的第一行包含一个整数 n (1 <= n <= 50),表示学生的个数,接下来的一行,包含 n 个整数,按顺序表示每个学生的能力值 ai(-50 <= ai <= 50)。接下来的一行包含两个整数,k 和 d (1 <= k <= 10, 1 <= d <= 50)。
输出一行表示最大的乘积。
3 7 4 7 2 50
49
while 1:
try:
n = int(raw_input())
A = map(int, raw_input().split())
k, d = map(int, raw_input().split())
except:
break
B = [[[0] * 2 for j in range(n)] for i in range(k + 1)]
for k in range(1, k + 1):
for i in range(n):
if k == 1 or i == 0:
B[k][i][0], B[k][i][1] = A[i], A[i]
else:
M = []
for t in range(1, d + 1):
if i - t < 0:
break
D = B[k - 1][i - t]
M.extend([D[0] * A[i], D[1] * A[i]])
B[k][i][0], B[k][i][1] = min(M), max(M)
print max([B[k][i][1] for i in range(n)])
#include<iostream>
#include<vector>
#include<algorithm>
using namespace std;
int main(){
int n;
while (cin >> n){
vector<int> a(n);
for (int i = 0; i < n; i++){
cin >> a[i];
}
int k, d;
cin >> k >> d;
vector<vector<long long>> dp_max(n, vector<long long>(k + 1, 0));
vector<vector<long long>> dp_min(n, vector<long long>(k + 1, 0));
for (int i = 0; i < n; i++){
dp_max[i][1] = a[i];
dp_min[i][1] = a[i];
}
for (int i = 0; i < n; i++){
for (int j = 2; j <= k; j++){
for (int m = max(0, i - d); m <= i - 1; m++){
dp_max[i][j] = max(dp_max[i][j], max(dp_max[m][j - 1] * a[i], dp_min[m][j - 1] * a[i]));
dp_min[i][j] = min(dp_min[i][j], min(dp_min[m][j - 1] * a[i], dp_max[m][j - 1] * a[i]));
}
}
}
long long max_value = dp_max[k - 1][k];
for (int i = k; i < n; i++){
max_value = max(max_value, dp_max[i][k]);
}
cout << max_value << endl;
}
return 0;
} import java.util.*;
5
-8 6 7 -7 9
3 50
| i=1时fm和fn只能取到k=1,i=1记录过程结果 | ||||||
| Max: | ||||||
| 0 | 1 | 2 | 3 | 4 | 5 | |
| 0 | 0 | 0 | 0 | 0 | 0 | 0 |
| 1 | 0 | -8 | 0 | 0 | 0 | 0 |
| 2 | 0 | 0 | 0 | 0 | 0 | 0 |
| 3 | 0 | 0 | 0 | 0 | 0 | 0 |
| Min: | ||||||
| 0 | 1 | 2 | 3 | 4 | 5 | |
| 0 | 0 | 0 | 0 | 0 | 0 | 0 |
| 1 | 0 | -8 | 0 | 0 | 0 | 0 |
| 2 | 0 | 0 | 0 | 0 | 0 | 0 |
| 3 | 0 | 0 | 0 | 0 | 0 | 0 |
| i=2时,fm和fn只能取到k=2,i=2保存结果 | ||||||
| Max: | ||||||
| 0 | 1 | 2 | 3 | 4 | 5 | |
| 0 | 0 | 0 | 0 | 0 | 0 | 0 |
| 1 | 0 | -8 | 6 | 0 | 0 | 0 |
| 2 | 0 | 0 | 0 | 0 | 0 | 0 |
| 3 | 0 | 0 | 0 | 0 | 0 | 0 |
| Min: | ||||||
| 0 | 1 | 2 | 3 | 4 | 5 | |
| 0 | 0 | 0 | 0 | 0 | 0 | 0 |
| 1 | 0 | -8 | 6 | 0 | 0 | 0 |
| 2 | 0 | 0 | -48 | 0 | 0 | 0 |
| 3 | 0 | 0 | 0 | 0 | 0 | 0 |
| i=3时,fm和fn根据之前保存结果计算结果 | ||||||
| Max: | ||||||
| 0 | 1 | 2 | 3 | 4 | 5 | |
| 0 | 0 | 0 | 0 | 0 | 0 | 0 |
| 1 | 0 | -8 | 6 | 7 | 0 | 0 |
| 2 | 0 | 0 | 0 | 42 | 0 | 0 |
| 3 | 0 | 0 | 0 | 0 | 0 | 0 |
| Min: | ||||||
| 0 | 1 | 2 | 3 | 4 | 5 | |
| 0 | 0 | 0 | 0 | 0 | 0 | 0 |
| 1 | 0 | -8 | 6 | 7 | 0 | 0 |
| 2 | 0 | 0 | -48 | -56 | 0 | 0 |
| 3 | 0 | 0 | 0 | -336 | 0 | 0 |
| i=4时,fm和fn根据之前保存结果计算结果 | ||||||
| Max: | ||||||
| 0 | 1 | 2 | 3 | 4 | 5 | |
| 0 | 0 | 0 | 0 | 0 | 0 | 0 |
| 1 | 0 | -8 | 6 | 7 | -7 | 0 |
| 2 | 0 | 0 | 0 | 42 | 56 | 0 |
| 3 | 0 | 0 | 0 | 0 | 392 | 0 |
| Min: | ||||||
| 0 | 1 | 2 | 3 | 4 | 5 | |
| 0 | 0 | 0 | 0 | 0 | 0 | 0 |
| 1 | 0 | -8 | 6 | 7 | -7 | 0 |
| 2 | 0 | 0 | -48 | -56 | -49 | 0 |
| 3 | 0 | 0 | 0 | -336 | -294 | 0 |
| i=5时,fm和fn根据之前保存结果计算结果 | ||||||
| Max: | ||||||
| 0 | 1 | 2 | 3 | 4 | 5 | |
| 0 | 0 | 0 | 0 | 0 | 0 | 0 |
| 1 | 0 | -8 | 6 | 7 | -7 | 9 |
| 2 | 0 | 0 | 0 | 42 | 56 | 63 |
| 3 | 0 | 0 | 0 | 0 | 392 | 504 |
| Min: | ||||||
| 0 | 1 | 2 | 3 | 4 | 5 | |
| 0 | 0 | 0 | 0 | 0 | 0 | 0 |
| 1 | 0 | -8 | 6 | 7 | -7 | 9 |
| 2 | 0 | 0 | -48 | -56 | -49 | -72 |
| 3 | 0 | 0 | 0 | -336 | -294 | -504 |
| | ||||||
| | | |||||
1、最大的乘积可能来自之前的最大乘积 * 当前的数值,也有可能来自之前的最小乘积 * 当前的数值,因此使用两个数组maxdp和mindp来记录当前位置处的最大和最小的乘积。
2、maxdp[i][j]表示,在必须以选择第i个同学为结尾的情况下选择j个同学的最大乘积。maxdp[i][j]的值可能来自:
max(maxdp[i-1][j-1]*arr[i], mindp[i-1][j-1]*arr[i])
max(maxdp[i-2][j-1]*arr[i], mindp[i-2][j-1]*arr[i])
...
max(maxdp[i-d][j-1]*arr[i], mindp[i-d][j-1]*arr[i])
选择最大的一个即可。
其中,maxdp[i-?][j-1]表示在i位置之前,以某一位置为结尾时选择j-1个同学的最大乘积,该位置需要满足两个条件:和j的距离不能超过d、该位置之前(包含该位置)必须至少有j-1个同学。
3、mindp与之类似。
#include<iostream>
#include<vector>
#include<algorithm>
#include<climits>
using namespace std;
long long process(vector<int> array,int n,int k, int d)
{
vector<long long> tmp(k+1, 0);
vector<vector<long long>> maxdp(n+1, tmp);
vector<vector<long long>> mindp(n+1, tmp);
for(int i=1; i<=n; i++)
{
maxdp[i][1] = array[i-1];
mindp[i][1] = array[i-1];
}
for(int i=2; i<=n; i++)
{
for(int j=2; j<=k && j<=i; j++) //总人数不能小于要选择的人数
{
long long maxPro = LLONG_MIN;
long long minPro = LLONG_MAX;
//和j的距离不能超过d且该位置之前(包含该位置)必须至少有j-1个同学
for(int l=max(j-1,i-d); l<=i-1; l++)
{
maxPro = max(maxPro, max(maxdp[l][j-1]*array[i-1], mindp[l][j-1]*array[i-1]));
minPro = min(minPro, min(maxdp[l][j-1]*array[i-1], mindp[l][j-1]*array[i-1]));
}
maxdp[i][j] = maxPro;
mindp[i][j] = minPro;
}
}
long long res = LLONG_MIN;
for(int i=1; i<=n; i++)
{
res = max(res, maxdp[i][k]);
}
return res;
}
int main()
{
int n;
cin >> n;
vector<int> array;
for(int i=0;i<n;i++)
{
int tmp;
cin >> tmp;
array.push_back(tmp);
}
int k,d;
cin >> k >>d;
cout << process(array,n,k,d);
system("pause");
return 0;
}
#include<stdio.h>
#include<vector>
#include<algorithm>
//#include<windows.h>
using namespace std;
struct node{
long Max,Min;
node():Max(0),Min(0){}
};
int a[100];
int main(){
int N,K,d,i,j,k;
//freopen("input.txt","r",stdin);
while(scanf("%d",&N)!=EOF){
for(i=0;i<N;i++) scanf("%d",&a[i]);
scanf("%d%d",&K,&d);
vector<vector<node> > dp(N,vector<node>(K+1));
long res=-999999999;
for(i=0;i<N;i++){
dp[i][1].Max=dp[i][1].Min=a[i];
res=max(res,dp[i][1].Max);
}
for(i=1;i<N;i++)
for(j=2;j<=min(K,i+1);j++){
for(k=1;k<=d&&k<=i;k++){
dp[i][j].Max=max(dp[i][j].Max,max(dp[i-k][j-1].Max*a[i],dp[i-k][j-1].Min*a[i]));
dp[i][j].Min=min(dp[i][j].Min,min(dp[i-k][j-1].Max*a[i],dp[i-k][j-1].Min*a[i]));
}
res=max(res,dp[i][j].Max);
}
printf("%ld\n",res);
}
} #python2 时间:35ms,内存:3060k 动态规划
length = input()
array = [int(x) for x in raw_input().split()]
k ,d =[int(x) for x in raw_input().split()]
array_max=array_min=array
for i in range(k-1):
new_max = [-float('inf')]*length
new_min = [float('inf')]*length
for num in range(i+1,length):
index_min = max(i,num-d)
index_max = min(num+d,length)
temp_max = -float('inf')
temp_min = float('inf')
for temp in range(index_min,num):
temp_max = max(temp_max,array[num]*array_max[temp],array[num]*array_min[temp])
temp_min = min(temp_min,array[num]*array_max[temp],array[num]*array_min[temp])
new_max[num]=temp_max
new_min[num]=temp_min
array_max = new_max[:]
array_min = new_min[:]
print(max(array_max))
import java.util.Scanner; public class Main { public static void main(String[] args) { Scanner sc = new Scanner(System.in); while (sc.hasNext()) { // 总人数 int n = sc.nextInt(); // 学生的能力值,第i个人对应第I个位置 int[] arr = new int[n + 1]; // 初始化 for (int i = 1; i <= n; i++) { arr[i] = sc.nextInt(); } // 选择学生数 int kk = sc.nextInt(); // 间距 int dd = sc.nextInt(); // 规划数组 long[][] f = new long[n + 1][kk + 1]; long[][] g = new long[n + 1][kk + 1]; // 初始化k=1的情况 for (int one = 1; one < n; one++) { f[one][1] = arr[one]; g[one][1] = arr[one]; } // 总人数等于1 或者大于1 if (n == 1) { System.out.println(arr[1]); } else { for (int k = 2; k <= kk; k++) { for (int one = k; one <= n; one++) { long tempmax = Long.MIN_VALUE; long tempmin = Long.MAX_VALUE; for (int left = Math.max(k - 1, one - dd); left <= one - 1; left++) { if (tempmax < Math.max(f[left][k - 1] * arr[one], g[left][k - 1] * arr[one])) { tempmax = Math.max(f[left][k - 1] * arr[one], g[left][k - 1] * arr[one]); } if (tempmin > Math.min(f[left][k - 1] * arr[one], g[left][k - 1] * arr[one])) { tempmin = Math.min(f[left][k - 1] * arr[one], g[left][k - 1] * arr[one]); } } f[one][k] = tempmax; g[one][k] = tempmin; } } long result = Long.MIN_VALUE; for (int one = kk; one <= n; one++) { if (result < f[one][kk]) { result = f[one][kk]; } } System.out.println(result); } } } }
比较明了的C++答案
#include <iostream>
#include <algorithm>
#include <vector>
#include <climits>
using namespace std;
int main(){
int n;
cin>>n;
vector<long long> a(n+1);
for(int i=1;i<=n;++i){
cin>>a[i];
}
int k,d;
cin>>k>>d;
vector<vector<long long>> dp_max(n+1,vector<long long>(k+1,0));
vector<vector<long long>> dp_min(n+1,vector<long long>(k+1,0));
long long res = LLONG_MIN;
for(int i=1;i<=n;++i){
dp_max[i][1] = dp_min[i][1] = a[i];
for(int j=2;j<=k&&j<=i;++j){
for(int l=1;l<=d&&i-l>=1;++l){
dp_max[i][j] = max(dp_max[i][j],max(dp_max[i-l][j-1]*a[i],dp_min[i-l][j-1]*a[i]));
dp_min[i][j] = min(dp_min[i][j],min(dp_max[i-l][j-1]*a[i],dp_min[i-l][j-1]*a[i]));
}
}
res = max(res,dp_max[i][k]);
}
cout<< res<<endl;
}
题目要求n各学生中选择k个,使这k个学生的能力值乘积最大。这是一个最优化的问题。另外,在优化过程中,提出了相邻两个学生的位置编号差不超过d的约束。
如果不用递归或者动态规划,问题很难入手,并且,限制条件d也需要对每一个进行约束,编程十分复杂
所以,解决的方法是采用动态规划(理由:1.求解的是最优化问题;2.可以分解为最优子结构)
1.对该问题的分解是关键。
从n个学生中,选择k个,可以看成是:先从n个学生里选择最后1个,然后在剩下的里选择k-1个,并且让这1个和前k-1个满足约束条件
2.数学描述
为了能够编程实现,需要归纳出其递推公式,而在写递推公式之前,首先又需要对其进行数学描述
记第k个人的位置为one,则可以用f[one][k]表示从n个人中选择k个的方案。然后,它的子问题,需要从one前面的left个人里面,选择k-1个,这里left表示k-1个人中最后一个(即第k-1个)人的位置,因此,子问题可以表示成f[left][k-1].
学生能力数组记为arr[n+1],第i个学生的能力值为arr[i]
one表示最后一个人,其取值范围为[1,n];
left表示第k-1个人所处的位置,需要和第k个人的位置差不超过d,因此
max{k-1,one-d}<=left<=one-1
在n和k定了之后,需要求解出n个学生选择k个能力值乘积的最大值。因为能力值有正有负,所以
当one对应的学生能力值为正时,
f[one][k] = max{f[left][k-1]arr[i]}(min{k-1,one-d}<=left<=one-1);
当one对应的学生能力值为负时
f[one][k] = max{g[left][k-1]arr[i]}(min{k-1,one-d}<=left<=one-1);
此处g[][]是存储n个选k个能力值乘积的最小值数组
遍历。因为一般看解答的多半是自己遇到问题不大会的,所以程序里面有写注释。如果大家不懂可以再问我,我回复或者再把解答写详细点。欢迎讨论。
import java.util.Scanner;
public class Main_jrh_AC {
public static void main(String[] args){
Scanner sc = new Scanner(System.in);
while(sc.hasNext()) {
//总人数
int n = sc.nextInt();
//学生能力值数组,第i个人直接对应arr[i]
int[] arr = new int[n + 1];
//初始化
for (int i = 1; i <= n; i++) {//人直接对应坐标
arr[i] = sc.nextInt();
}
//选择的学生数
int kk = sc.nextInt();
//间距
int dd = sc.nextInt();
/**
* 递推的时候,以f[one][k]的形式表示
* 其中:one表示最后一个人的位置,k为包括这个人,一共有k个人
* 原问题和子问题的关系:f[one][k]=max{f[left][k-1]*arr[one],g[left][k-1]*arr[one]}
*/
//规划数组
long[][] f = new long[n + 1][kk + 1];//人直接对应坐标,n和kk都要+1
long[][] g = new long[n + 1][kk + 1];
//初始化k=1的情况
for(int one = 1;one<=n;one++){
f[one][1] = arr[one];
g[one][1] = arr[one];
}
//自底向上递推
for(int k=2;k<=kk;k++){
for(int one = k;one<=n;one++){
//求解当one和k定的时候,最大的分割点
long tempmax = Long.MIN_VALUE;
long tempmin = Long.MAX_VALUE;
for(int left = Math.max(k-1,one-dd);left<=one-1;left++){
if(tempmax<Math.max(f[left][k-1]*arr[one],g[left][k-1]*arr[one])){
tempmax=Math.max(f[left][k-1]*arr[one],g[left][k-1]*arr[one]);
}
if(tempmin>Math.min(f[left][k-1]*arr[one],g[left][k-1]*arr[one])){
tempmin=Math.min(f[left][k-1]*arr[one],g[left][k-1]*arr[one]);
}
}
f[one][k] = tempmax;
g[one][k] = tempmin;
}
}
//n选k最大的需要从最后一个最大的位置选
long result = Long.MIN_VALUE;
for(int one = kk;one<=n;one++){
if(result<f[one][kk]){
result = f[one][kk];
}
}
System.out.println(result);
}
}
}
#include<iostream>
using namespace std;
#define MAX 100
inline long long max(long long a,long long b){return a>b?a:b;}
inline long long min(long long a,long long b){return a>b?b:a;}
int main()
{
int N,K,D;
int num[MAX]={0};
long long pMax[MAX][MAX]={0};
long long pMin[MAX][MAX]={0};
cin >> N;
for(int i=1;i<=N;i++)
cin >> num[i];
cin>>K>>D;
//思路造成我的k值在最外循环
for(int k=1;k<=K;k++)
{
for(int i=1;i<=N;i++)
{
//思路的差异在初始值上也不同
if(k==1)
pMax[k][i]=pMin[k][i]=num[i];
else
{
for(int j=i-1;i-j<=D&&j>0;--j)
{
pMax[k][i]=max(pMax[k][i],max(pMax[k-1][j]*num[i],pMin[k-1][j]*num[i]));
pMin[k][i]=min(pMin[k][i],min(pMax[k-1][j]*num[i],pMin[k-1][j]*num[i]));
}
}
}
}
long long result = 0;
//得出最后的结果
for(int i=1;i<=N;i++)
result = max(result,pMax[K][i]);
cout << result;
return 0;
}
- import java.util.Scanner; //主要参考@菜鸡华的代码 //kk dd one命名引起不适所以替换成简洁有力的命名K D i //在N的遍历中使其终止与N-K+k,不进行剩下不必要的训练。 public class Main{ public static void main(String[] args){ Scanner sc = new Scanner(System.in); while(sc.hasNext()) { //读数据 int N = sc.nextInt(); int[] arr = new int[N + 1]; for (int i = 1; i <= N; i++) { arr[i] = sc.nextInt(); } int K = sc.nextInt(); int D = sc.nextInt(); //规划数组表示,使用两个数组保证,最大正数最小负数都有保存 long[][] f = new long[N + 1][K + 1];//人直接对应坐标,n和K都要+1 long[][] g = new long[N + 1][K + 1]; //初始化k=1的情况 for(int i = 1;i<=N;i++){ f[i][1] = arr[i]; g[i][1] = arr[i]; } //自底向上递推 for(int k=2;k<=K;k++){ for(int i = k;i<=N-K+k;i++){ //求解当i和k定的时候,每步最优子问题 long tempmax = Long.MIN_VALUE; long tempmin = Long.MAX_VALUE; for(int left = Math.max(k-1,i-D);left<=i-1;left++){ if(tempmax<Math.max(f[left][k-1]*arr[i],g[left][k-1]*arr[i])){ tempmax=Math.max(f[left][k-1]*arr[i],g[left][k-1]*arr[i]); } if(tempmin>Math.min(f[left][k-1]*arr[i],g[left][k-1]*arr[i])){ tempmin=Math.min(f[left][k-1]*arr[i],g[left][k-1]*arr[i]); } } f[i][k] = tempmax; g[i][k] = tempmin; } } //从最后一行选取最终最大值 long result = Long.MIN_VALUE; for(int i = K;i<=N;i++){ if(result<f[i][K]){ result = f[i][K]; } } System.out.println(result); } } }
N = int(raw_input())
A = map(int, raw_input().split())
K, D = map(int, raw_input().split())
#因为有正反所以分成两部分
dpmax = [[0]*N for i in range(K)]#列出未来将要用到的结果位置
dpmin = [[0]*N for i in range(K)]#同上
res = (1<<64)*-1#默认的最小值,后面比较需要
for n in range(N):
dpmax[0][n] = dpmin[0][n] = A[n]#就是在k=1时,取第一个值时
for k in range(1,K):
for pre in range(max(0,n-D),n):#因为有间隔,所以如果间隔大于n,可以全取,反之需从间隔部分之后开始
dpmax[k][n] = max(dpmax[k][n],max(dpmax[k-1][pre]*A[n],dpmin[k-1][pre]*A[n]))#不断与之前的进行比较获取最大
dpmin[k][n] = min(dpmin[k][n],min(dpmax[k-1][pre]*A[n],dpmin[k-1][pre]*A[n]))#不断与之前进行比较获取最小
res = max(res,dpmax[K-1][n])
#其实结果可以写成max([x for x in dpmax[K-1]]))
#因为是要在dpmax[K-1]取出最大的那个
print(max([x for x in dpmax[K-1]])))
#include <bits/stdc++.h>
using namespace std;
void search(vector<vector<vector<long>>>& ***, const vector<int>& a, int m_d, int i, int k, int c_d, long product, long& max)
{
// c_d代表与上一个被选中同学的距离
if (k == 0)
{
if (product > max)
max = product;
return;
}
// 与上一个同学的距离已经超过了d,不符合要求
if (c_d >= m_d)
return;
if (i >= a.size())
return;
// 把剩下所有的同学选了也达不到人数要求,直接跳过
if (i + k > a.size())
return;
// 以下的判断用于缓存数据,减少不必要的判断。不用缓存只能过80%。
if (abs(product) >= ***[i][k][c_d])
***[i][k][c_d] = abs(product);
else
return;
// 选中i
search(***, a, m_d, i + 1, k - 1, 0, product * a[i], max);
// 不选中i
search(***, a, m_d, i + 1, k, product == 1 ? 0 : c_d + 1, product, max);
}
int main() {
int n;
cin >> n;
vector<int> a(n);
for (int i = 0; i < n; i++)
{
cin >> a[i];
}
int k, d;
cin >> k >> d;
vector<vector<vector<long>>> ***(n, vector<vector<long>>(k + 1, vector<long>(d, 0)));
long max = 0;
search(***, a, d, 0, k, 0, 1, max);
cout << max << endl;
}
假设已经确定了k-1个人,那么只要从接下来的d个人中找到使乘积最大的人添加。反过来即得到递推思路,我们以i位置结尾,计算当前的最大乘积dp[i][k]是在j属于i-d到i之间的这部分dp[j][k-1]的基础上考虑的,也就是说一次只考虑d变化。
public class Main{
public static void main(String[] args) {
Scanner scanner = new Scanner(System.in);
int N = scanner.nextInt();
int stu[] = new int[N];
for (int i = 0; i < stu.length; i++) {
stu[i] = scanner.nextInt();
}
int K = scanner.nextInt();
int D = scanner.nextInt();
getMaxProduct(N, stu, K, D);
}
/**
* 获取最大乘积
* 从n个学生中,选择k个,可以看成是:先从n个学生里选择最后1个,然后在剩下的里选择k-1个,并且让这1个和前k-1个满足约束条件
*/
private static void getMaxProduct(int N, int[] stu, int K, int D) {
// 记录以当前位置结束的取1-k个值时的最大乘积, 分正负两个情况
long fmax[][] = new long[N][K];
long fmin[][] = new long[N][K];
for (int i = 0; i < N; i++) {
// 第一个人
fmax[i][0] = stu[i];
fmin[i][0] = stu[i];
// 添加其他人
for(int k=1; k<K; k++){
// 每次只考虑添加一个人,那么范围缩小到d了!
for(int j=i-1; j>=0 && i-j<=D; j--){
// 是否替换一定是跟i和问题(乘积)有关的,并且和i-d~i范围的比较
fmax[i][k] = Math.max(fmax[i][k], Math.max(fmax[j][k-1]*stu[i], fmin[j][k-1]*stu[i]));
fmin[i][k] = Math.min(fmin[i][k], Math.min(fmax[j][k-1]*stu[i], fmin[j][k-1]*stu[i]));
}
}
}
// 取最小值(负数)只可能存在中间过程,最后结果不管正负都是取最大的那个,所以只在fmax中找
long max = fmax[0][K-1];
for (int i = 1; i < N; i++) {
max = Math.max(max, fmax[i][K-1]);
}
System.out.println(max);
}
}
#include<stdio.h>
#include<stdlib.h>
#include<vector>
#include<iostream>
using namespace std;
int main() {
int n,k,d,temp;
int maximum=0;
vector<int> a;
cin>>n;
for(int yy=0;yy<n;yy++) {
cin>>temp;
a.push_back(temp);
}
cin>>k;
cin>>d;
int dpm[k][n];
int dpn[k][n];
for(int x=0;x<k;x++) {
for(int y=0;y<n;y++) {
dpm[x][y]=0;
dpn[x][y]=0;
}
}
for(int xx=0;xx<n;xx++) {
dpm[0][xx]=a[xx];
dpn[0][xx]=a[xx];
}
for(int i=0;i<n;i++) {
for(int kk=1;kk<k;kk++) {
for(int j=i-1;j>max(-1,i-d-1);j--) {
dpm[kk][i]=max(dpm[kk][i],max(dpm[kk-1][j]*a[i],dpn[kk-1][j]*a[i]));
dpn[kk][i]=min(dpn[kk][i],min(dpm[kk-1][j]*a[i],dpn[kk-1][j]*a[i]));
}
}
maximum=max(maximum,dpm[k-1][i]);
}
/*for(int xxx=k;xxx<n;xxx++) {
maximum=max(maximum,dpm[k-1][xxx]);
}*/
cout<<maximum<<endl;
return 0;
}
这个代码一直卡在50%通过率,
47
-41 -5 -10 -31 -44 -16 -3 -33 -34 -35 -44 -44 -25 -48 -16 -32 -37
-8 -33 -30 -6 -18 -26 -37 -40 -30 -50 -32 -5 -41 -32 -12 -33 -22 -14 -34
-1 -41 -45 -8 -39 -27 -23 -45 -10 -50 -34
6 3
这个例子上,正确的输出应该是5504557500
我的输出为:
2135936000
/*背包变形 + 状态压缩*/
#include <bits/stdc++.h>
using namespace std;
int main(int argc, char *argv[])
{
long dp[50][2], res;
int a[50], n, k, d, i, j, delta;
scanf("%d", &n);
for (i = 0; i < n; ++i)
{
scanf("%d", a + i);
dp[i][0] = dp[i][1] = a[i];
}
scanf("%d%d", &k, &d);
for (i = 1; i < k; ++i)
{
for (j = n - 1; j >= i; --j)
{
for (delta = 1; delta <= d && j >= delta; ++delta)
{
dp[j][0] = min(dp[j][0], min(dp[j - delta][0] * a[j], dp[j - delta][1] * a[j]));
dp[j][1] = max(dp[j][1], max(dp[j - delta][0] * a[j], dp[j - delta][1] * a[j]));
}
}
}
for (res = dp[k - 1][1], i = k; i < n; ++i)
{
if (dp[i][1] > res)
{
res = dp[i][1];
}
}
printf("%ld\n", res);
return 0;
}