首页 > 试题广场 >

合唱团

[编程题]合唱团
  • 热度指数:104953 时间限制:C/C++ 1秒,其他语言2秒 空间限制:C/C++ 32M,其他语言64M
  • 算法知识视频讲解
有 n 个学生站成一排,每个学生有一个能力值,牛牛想从这 n 个学生中按照顺序选取 k 名学生,要求相邻两个学生的位置编号的差不超过 d,使得这 k 个学生的能力值的乘积最大,你能返回最大的乘积吗?

输入描述:
每个输入包含 1 个测试用例。每个测试数据的第一行包含一个整数 n (1 <= n <= 50),表示学生的个数,接下来的一行,包含 n 个整数,按顺序表示每个学生的能力值 ai(-50 <= ai <= 50)。接下来的一行包含两个整数,k 和 d (1 <= k <= 10, 1 <= d <= 50)。


输出描述:
输出一行表示最大的乘积。
示例1

输入

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)])


发表于 2017-08-19 23:21:33 回复(4)
思路和大家差不多,动态规划问题。dp_max[i][j],dp_min[i][j]分别表示以第i个人位结尾,选择j个人的最大乘积和最小乘积
#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;
}

发表于 2017-08-16 11:42:59 回复(4)
0. 每一个状态只与前一个状态有关
1. 前一个状态是一个范围,前D个之内的一个
2. 有正负所以需要维护两个表
N = int(raw_input().strip())
ab = map(int, raw_input().strip().split(' '))
(K, D) = map(int, raw_input().strip().split(' '))

dmax = [[0] * N for i in range(K)]
dmin = [[0] * N for i in range(K)]
res = (1 << 64) * -1

for i in range(N):
    dmax[0][i] = dmin[0][i] = ab[i] \
    for k in range(1, K):
        for pre in range(max(0, i - D), i):
            dmax[k][i] = max(dmax[k][i],
                max(dmax[k - 1][pre] * ab[i],dmin[k - 1][pre] * ab[i]))
            dmin[k][i] = min(dmin[k][i],    min(dmax[k - 1][pre] * ab[i], dmin[k - 1][pre] * ab[i]))
    res = max(res, dmax[K - 1][i])

print res


编辑于 2018-03-06 18:50:41 回复(0)
……吐槽一下,为什么先我回答的2位写的那么长……

f [ i ] [ j ] [ 最大 / 最小 ]

分别表示,以第i个人为最后一个(也是必选的)人,加上这个人,已经选了 j 个人,最大可能的乘积和最小可能的乘积。

#include <stdio.h>
#include <ctype.h>
#include <string.h>
#include <stdlib.h>
#include <limits.h>
#include <math.h>
#include <algorithm>
using namespace std;
typedef long long ll;

int a[55];
ll f[55][15][2];

int main(){
    int n,kk,d;
    scanf("%d",&n);
    for(int i=1;i<=n;++i){
        scanf("%d",&a[i]);
    }
    scanf("%d%d",&kk,&d);
    ll ans=0;
    for(int i=1;i<=n;i++){
        f[i][1][0]=f[i][1][1]=a[i];
        for(int j=2;j<=kk;++j){
            for(int k=i-1;k>=max(i-d,1);--k){
                f[i][j][0]=max(f[i][j][0],max(f[k][j-1][0]*a[i],f[k][j-1][1]*a[i]));
                f[i][j][1]=min(f[i][j][1],min(f[k][j-1][0]*a[i],f[k][j-1][1]*a[i]));
            }
        }
        ans=max(ans,max(f[i][kk][0],f[i][kk][1]));
    }
    printf("%lld\n",ans);
    return 0;
}
发表于 2016-08-08 22:35:53 回复(16)
 import java.util.*;
public class Main{
    
    /**
    总共n个学生,从中选k个学生计算乘积
    f(i,k)代表选出k个学生,最后一个学生编号是i.. i从0到n-1
    最终在f(0,k),f(1,k),f(2,k)...f(n-1,k)中选取最大的乘积作为结果
    
    所以现在要解决的问题是怎么求f(i,k)
    假设第k-1个学生的编号为j,f(j,k-1)已知
    那么f(i,k)就可以用到f(j,k-1)的条件了。但由于学生的能力值可以为负数。
    若第i个学生的能力值为负数,要使乘积最大,则应求f(j,k-1)的最小值,记为Min.f(j,k-1);
    若第i个学生的能力值为正数,要使乘积最大,则应求f(j,k-1)的最大值,记为Max.f(j,k-1)。
    因此应该设立两个数组同时保存选取第k-1个学生后乘积的最大和最小值。
    
    values(i)代表第i个学生的能力值
    状态方程为:f(i,k) = max(f(i,k), Min.f(j,k-1)*values(i), Max.f(j,k-1)*values(i))
    
    由于i和j满足条件 i-j <= d,所以在满足条件的j中选取最大的值作为f(i,k)的值
    **/
    
    public static long getMax(long a, long b, long c){
        return Math.max(Math.max(a,b),c);
    }
    
    public static long getMin(long a, long b, long c){
        return Math.min(Math.min(a,b),c);
    }
    
    public static void main(String[] args){
        Scanner scanner = new Scanner(System.in);
        int sum =1;
        int n = Integer.parseInt(scanner.nextLine());
        int[] values = new int[n];
        for(int i=0; i <n; i++){
            values[i] = Integer.parseInt(scanner.next());
        }
        int k = Integer.parseInt(scanner.next());
        int d = Integer.parseInt(scanner.next());
        
        long[][] max = new long[n][k];
        long[][] min = new long[n][k];
        
        for(int i=0; i<n; i++){
            for(int j=0; j<k; j++){
                max[i][j] = min[i][j] = 0;
            }
        }
        
        for(int i=0; i<n; i++){
            max[i][0] = min[i][0] = values[i];
        }
        
        for(int j=1; j<k; j++){
            for(int i=0; i<n; i++){
                if(i==0) continue;
                for(int m=i-1; i-m <= d; m--){
                    if(m<0) break;
                    max[i][j] = getMax(max[i][j], max[m][j-1]*values[i], min[m][j-1]*values[i]);
                    min[i][j] = getMin(min[i][j], max[m][j-1]*values[i], min[m][j-1]*values[i]);
                }
            }
        }
        long ans = 0;
        for(int i=0; i<n; i++){
            ans = Math.max(ans, max[i][k-1] );
        }
        System.out.print(ans);
    } 
}

发表于 2018-02-15 15:31:57 回复(1)
这里借用ZhanHeng的代码,我只是打印出了过程
例如

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



具体推到就是:先找子问题的最优,并记录结果,递推
fmax[k][i] = Math.max(fmax[k][i], Math.max(fmax[k - 1][j] * arr[i], fmin[k - 1][j] * arr[i]));
fmin[k][i] = Math.min(fmin[k][i], Math.min(fmax[k - 1][j] * arr[i], fmin[k - 1][j] * arr[i]));
发表于 2017-08-20 10:47:07 回复(0)
import java.util.*;
public class Main{
    public static void main(String[] args){
        Scanner scan = new Scanner(System.in);
        int n = scan.nextInt();
        int[] nums = new int[n];
        for(int i = 0; i < n; i++){
            nums[i] = scan.nextInt();
        }
        int k = scan.nextInt();
        int d = scan.nextInt();
        long[][] max = new long[k][n];
        long[][] min = new long[k][n];
        for(int i = 0; i < k; i++)
            for(int j = 0; j < n; j++){
            	//min[i][j] = Integer.MAX_VALUE;
            	max[i][j] = 1;
            	if(i == 0){
                    min[i][j] = nums[j];
                    max[i][j] = nums[j];
                }
        }
        
        for(int i = 1; i < k; i++)
            for(int j = 0; j < n; j++)
            	for(int m = 1; m <= d; m++){
            	if(j - m >= 0){
                    if(nums[j] > 0){
            		min[i][j] = Math.min(min[i][j], min[i - 1][j - m] * nums[j]);
            		max[i][j] = Math.max(max[i][j], max[i - 1][j - m] * nums[j]);
                    } else{
                       min[i][j] = Math.min(min[i][j], max[i - 1][j - m] * nums[j]);
                       max[i][j] = Math.max(max[i][j], min[i - 1][j - m] * nums[j]);
                    }
                }
        }
        long Max = 0;
        for(int i = 0; i < n; i++)
            Max = Math.max(Max, max[k - 1][i]);
        System.out.println(Max);

    }
}

发表于 2016-08-27 20:22:37 回复(4)

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;
}
发表于 2018-03-20 11:30:12 回复(1)
#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);
	}
}

发表于 2017-08-10 19:19:15 回复(1)
这道题之所以比较难以看出来就是多套了一层;假定这道题这样问说,以每个位置结尾所能取到的最大值;也就是求f(n) ,以位置n结尾的最大值,位置n为结尾想得到最大值,那就肯定依赖于前面;然后我们想说依赖于前面也就是d个数,然后再往前,也就是暴力递归的做法,但是我们还有一个要求就是k个数,也就是如果f(n,k) 想是最大值,f((n-d~n),k-1) 就必须是最大值,因为我们以n为结尾,后面不用考虑,那必须就是前面是最大值;然后以每个位置为结尾的最大值得到了,然后都比较一遍,就可以得到最好的选择了;
看来都是套路。。 大家要记得以每个位置结尾这种思维方式哦;
发表于 2017-08-09 19:35:48 回复(7)
#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))

发表于 2017-10-21 18:41:03 回复(0)
#include <iostream>
#include <fstream>
using namespace std;

int max(int a, int b){
return a > b ? a : b;
}
void swap(int &a, int &b){
int c = a;
a = b;
b = c;
}

int main()
{
int n, k, d;
cin >> n;
int *val = new int[n + 1];
for (int i = 1; i <= n; ++i){
cin >> val[i];
}
cin >> k >> d;

//初始化
long long **arr_up = new long long*[k + 1];
long long **arr_down = new long long*[k + 1];
for (int i = 0; i < k + 1; ++i){
arr_up[i] = new long long[n + 1]();
arr_down[i] = new long long[n + 1]();
}
for (int i = 1; i < n + 1; ++i){
arr_up[1][i] = val[i];
arr_down[1][i] = val[i];
}

for (int i = 2; i <= k; ++i){ //选i个人
for (int j = i; j <= n; ++j){ //最大的位置编号
int x = max(j - d, 1);
long long sum_up = arr_up[i - 1][x] * val[j];//记录最大的数
long long sum_down = arr_down[i - 1][x] * val[j];//记录最小的数
if (sum_up < sum_down){
swap(sum_up, sum_down);
}
for (++x; x < j; ++x){
long long num = arr_up[i - 1][x] * val[j];
if (num > sum_up){
sum_up = num;
}
else if (num < sum_down){
sum_down = num;
}

num = arr_down[i - 1][x] * val[j];
if (num > sum_up){
sum_up = num;
}
else if (num < sum_down){
sum_down = num;
}
}
arr_up[i][j] = sum_up;
arr_down[i][j] = sum_down;
}
}
//获取最大乘积
long long max_sum = arr_up[k][k];
for (int i = k + 1; i <= n; ++i){
if (arr_up[k][i] > max_sum){
max_sum = arr_up[k][i];
}
}
cout << max_sum << endl;

//system("pause");
return 0;
}
发表于 2017-09-07 22:16:21 回复(0)
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);
			}

		}
	}

} 


发表于 2017-08-29 10:29:33 回复(0)

比较明了的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;
}
发表于 2017-08-10 12:04:25 回复(1)

网易_合唱团解析

1. 题目分析

题目要求n各学生中选择k个,使这k个学生的能力值乘积最大。这是一个最优化的问题。另外,在优化过程中,提出了相邻两个学生的位置编号差不超过d的约束。

如果不用递归或者动态规划,问题很难入手,并且,限制条件d也需要对每一个进行约束,编程十分复杂

所以,解决的方法是采用动态规划(理由:1.求解的是最优化问题;2.可以分解为最优子结构)

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个能力值乘积的最小值数组

3.编程实现

遍历。因为一般看解答的多半是自己遇到问题不大会的,所以程序里面有写注释。如果大家不懂可以再问我,我回复或者再把解答写详细点。欢迎讨论。

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);
        }
    }
}
编辑于 2017-09-09 23:01:06 回复(41)
我和大多数人的算法略有不同,在子问题分解上我才用从n个学生中选取1人、2人...直至k人的思路,而不是先从n个学生里选择最后1个,然后在剩下的里选择k-1个,这样的思路在编程上会造成循环上的调整和最后结果的比较不同,算法已通过样本,理解该算法最好方式是手动演算一下计算过程。



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

编辑于 2019-03-28 16:39:54 回复(0)
对于这道题我的具体思路:
  1. 看到这道题,首先最基本的想法一般是从n中抽取k个数,然后验证是否符合条件,复杂度大概是是一个非指数级可解问题
  2. 接着进一步优化,首先可以发现因为限制了相对位置d,所以问题是位置相关的,而根据乘法交换率,计算时位置无关,因此保持其位置而不是考虑排序方法更好归纳问题。因此设人的序列为ListA:a1,a2,....,an
  3. 这道题主要超变量有ListA,N.K,D从这道题跟具体的人选取与不选取有关,从头或从尾砍掉或选取一部分人,可以保持这个问题本身性质不变而只在超变量有改变,具有子问题的性质,且是最优化问题且包含最优子问题,因此考虑上贪心算法或者动态规划算法,并且先不考虑正负问题。
  4. 贪心的2个要求是最优子问题和贪心选择(在子问题迭代过程每一步都是最佳,其中不会有其它选择优于我的选择【这也是很多贪心算法的证明】)考虑ListA中对an这个人的选择入k和抛弃,前面子问题会变成k-1或者k,显而易见直接丢掉算子问题不能保证最优,因此考虑动态规划。
  5. 动态规划要求是最优子问题,重叠子问题,而动态规划又经常被大牛叫为打表算法,因此考虑打表,动态规划首先都要列出递推方程,考虑以砍尾部来划归子问题
  6. (图里公式max的第一行d应该为大写D,其应该被重置)
  7. 但是这样就会被列为三维矩阵,非常麻烦且不直观,考虑到距离d会因为选择了某人而重置为D所以考虑另外一种切分方法
  8. 但是这种方法有个问题就是listA前面的后面可能会空一大截,不一定从尾巴或头部开始切,这也符合实际问题需要。于是表大概就长下面这个样子:
  9. 这样就可以解决最大问题了,但是因为涉及正负数,负数最小的可能在下一次乘法中反成最大,所以得考虑打两个表,一个最小表一个最大表,判断哪个大取哪个。
  10. 照这个思路的具体实现跟高赞答案一样
  1. 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);         }     } }

发表于 2018-12-25 16:55:19 回复(1)

参考各位大佬的解答如:bupt渣硕,加深自己对动态规划问题的理解,附上代码和理解注释

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]])))

动态规划问题差不多一次看见,得多找一些练习一下୧(๑•̀⌄•́๑)૭

发表于 2018-09-23 14:48:59 回复(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;
}
发表于 2018-09-16 01:45:09 回复(0)

思路

设dpMax = [[1 for i in range(k+1)] for j in range(n+1)] 表示以第i个人为最后一个人,一共选取了j个人时的最大乘积。 同理,dpMin = [[1 for i in range(k+1)] for j in range(n+1)] 表示同样状态下的最小乘积(由于数据中存在负数,负数乘上某个极大的负数反而会变成正的极大值,因而需要同时记录最小值)。 dpMax[i][j] 很显然与dpMax[p][j-1] (for p in range(max(i-d, 0), i)表示遍历当前数之前符合范围条件的所有数, 而 j-1则表示所有数中每一个数对应的只比当前乘积因子数少一个时的结果)相关,可以理解为dpMax[i][j]由两部分组成,一部分是自身作为待选值,另一部分是dpMax[p][j-1]乘上一个人后得到的值,然后取它们的极大值,由此可以得到状态转移方程如下: dpMax[i][j] = max(dpMax[i][j], max(dpMax[p][j-1]array[i-1], dpMin[p][j-1]array[i-1])) dpMin[i][j] = min(dpMin[i][j], min(dpMax[p][j-1]array[i-1], dpMin[p][j-1]array[i-1]))

n = int(input()) array = [int(x) for x in input().split()] k, d =[int(x) for x in input().split()] dpMax = [[1 for i in range(k+1)] for j in range(n+1)] dpMin = [[1 for i in range(k+1)] for j in range(n+1)] result = 0 for i in range(1, n+1): for j in range(1, k+1): for p in range(max(i-d, 0), i): dpMax[i][j] = max(dpMax[i][j], max(dpMax[p][j-1]*array[i-1], dpMin[p][j-1]*array[i-1])) dpMin[i][j] = min(dpMin[i][j], min(dpMax[p][j-1]*array[i-1], dpMin[p][j-1]*array[i-1])) if dpMax[i][j] > result: result = dpMax[i][j] print(result)

编辑于 2018-09-08 10:40:28 回复(1)

问题信息

难度:
185条回答 71681浏览

热门推荐

通过挑战的用户

查看代码