动态规划-选学生

该博客讨论了一种动态规划方法,用于解决从n个学生中按照顺序选取k名学生的问题,要求相邻学生的编号差不超过d,使得这k个学生的能力值乘积最大。博主详细阐述了思路,包括考虑正负值、状态转移方程,并给出了读取数据、动态分配内存的实现。最后通过不断更新状态找到最大乘积。
摘要由CSDN通过智能技术生成

题目

有 n 个学生站成一排,每个学生有一个能力值,
从这 n 个学生中按照顺序选取 k 名学生,
要求相邻两个学生的位置编号的差不超过 d,
使得这 k 个学生的能力值的乘积最大,你能返回最大的乘积吗

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

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


思路

  1. 因为有正有负,所以需要两个数组,最大最小,Fmax,Fmin K*N维的,K是选几人,N是总共所少人
  2. 读数据,动态分配释放内存
  3. Fmax[k][i]表示以i个人结束,肯定要选,然后已选了k个,下标从0开始和对应人数的要-1
  4. k=0时,表示只选1个人,那么就是i: 0~n-1总选最大的,初始化过程
  5. k: 1~K-1, 与之前的状态k-1有关,然后最大间隔是D, 所以间隔m: 1~D,学生位数就是i-m,当i-m>=0Fmax[k][i] = max(Fmax[k][i], max(Fmax[k-1][i-m]*ai[i],Fmin[k-1][i-m]*ai[i]));同理,Fmin同样计算,但是是取小的
  6. 不断更新,k++
  7. 最后就从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 [N]();  //初始化为0;加上()
        Fmin[x] = new long long [N]();
        // 初始化,选一个人的时候,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][n]);
    }

    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[n];

        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;
}
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值