洛谷P1018乘积最大——区间DP

题目:https://www.luogu.org/problemnew/show/P1018

区间DP+高精,注意初始化和转移的细节。

代码如下:

#include<iostream>
#include<cstdio>
#include<cstring>
#define MAXN 20005
using namespace std;
typedef long long ll;
ll n,k,a[45],f[45][7][MAXN],tmp[MAXN],num[MAXN];
char dc[45];
void nm(ll l,ll r)
{
//    printf("l=%lld r=%lld\n",l,r);
    num[0]=r-l+1;
    for(ll i=l;i<=r;i++)//正存即可
        num[i-l+1]=a[i];
//    for(int i=1;i<=num[0];i++)printf("%lld",num[i]);
//    cout<<endl;
}
void init()
{
    a[0]=strlen(dc);
    for(ll i=1;i<=a[0];i++)
    {
        a[i]=dc[a[0]-i]-‘0‘;
//        memset(num,0,sizeof num);
//        nm(1,i);
        memcpy(f[i][0],f[i-1][0],sizeof f[i-1][0]);//注意初始化!
        f[i][0][0]++;f[i][0][i]=a[i];
    }
}
void ch(ll a[],ll b[])
{
    tmp[0]=a[0]+b[0]-1;
    for(ll i=1;i<=a[0];i++)
        for(ll j=1;j<=b[0];j++)
        {
            tmp[i+j-1]+=a[i]*b[j];
            tmp[i+j]+=tmp[i+j-1]/10;
            tmp[i+j-1]%=10;
        }
    if(tmp[tmp[0]+1])tmp[0]++;
}
bool com(ll a[],ll b[])
{
    if(a[0]>b[0])return 1;
    if(a[0]<b[0])return 0;
    for(ll i=a[0];i;i--)
    {
        if(a[i]>b[i])return 1;
        if(a[i]<b[i])return 0;
    }
    return 0;
}
void print(ll a[])
{
    for(ll i=a[0];i;i--)
        printf("%lld",a[i]);
}
int main()
{
    scanf("%lld%lld",&n,&k);
    cin>>dc;
    init();
//    f[1][1][0]=1;f[1][1][1]=1;
    for(ll i=2;i<=n;i++)
        for(ll l=1;l<=min(i-1,k);l++)
            for(ll j=l;j<i;j++)//从l开始!!!
            {
//                f[i][l]=max(f[j][l-1]*num(j+1,i),f[i][l]);
                memset(tmp,0,sizeof tmp);
                memset(num,0,sizeof num);
                nm(j+1,i);
                ch(f[j][l-1],num);
                if(com(tmp,f[i][l]))memcpy(f[i][l],tmp,sizeof tmp);
            }
    print(f[n][k]);
    return 0;
}

原文地址:https://www.cnblogs.com/Zinn/p/8533365.html

时间: 2024-10-10 10:50:50

洛谷P1018乘积最大——区间DP的相关文章

洛谷P1018 乘积最大 区间动归

题目描述 今年是国际数学联盟确定的"2000――世界数学年",又恰逢我国著名数学家华罗庚先生诞辰90周年.在华罗庚先生的家乡江苏金坛,组织了一场别开生面的数学智力竞赛的活动,你的一个好朋友XZ也有幸得以参加.活动中,主持人给所有参加活动的选手出了这样一道题目: 设有一个长度为N的数字串,要求选手使用K个乘号将它分成K+1个部分,找出一种分法,使得这K+1个部分的乘积能够为最大. 同时,为了帮助选手能够正确理解题意,主持人还举了如下的一个例子: 有一个数字串:312, 当N=3,K=1时

洛谷P1220关路灯——区间DP

题目:https://www.luogu.org/problemnew/show/P1220 区间DP. 代码如下: #include<iostream> #include<cstdio> using namespace std; int n,c,pos[55],w[55],sum,s[55][55],dp[55][55][3],INF=10000006; int main() { scanf("%d%d",&n,&c); for(int i=1

洛谷P1220关路灯[区间DP]

题目描述 某一村庄在一条路线上安装了n盏路灯,每盏灯的功率有大有小(即同一段时间内消耗的电量有多有少).老张就住在这条路中间某一路灯旁,他有一项工作就是每天早上天亮时一盏一盏地关掉这些路灯. 为了给村里节省电费,老张记录下了每盏路灯的位置和功率,他每次关灯时也都是尽快地去关,但是老张不知道怎样去关灯才能够最节省电.他每天都是在天亮时首先关掉自己所处位置的路灯,然后可以向左也可以向右去关灯.开始他以为先算一下左边路灯的总功率再算一下右边路灯的总功率,然后选择先关掉功率大的一边,再回过头来关掉另一边

[NOIP2000] 提高组 洛谷P1018 乘积最大

题目描述 今年是国际数学联盟确定的“2000――世界数学年”,又恰逢我国著名数学家华罗庚先生诞辰90周年.在华罗庚先生的家乡江苏金坛,组织了一场别开生面的数学智力竞赛的活动,你的一个好朋友XZ也有幸得以参加.活动中,主持人给所有参加活动的选手出了这样一道题目: 设有一个长度为N的数字串,要求选手使用K个乘号将它分成K+1个部分,找出一种分法,使得这K+1个部分的乘积能够为最大. 同时,为了帮助选手能够正确理解题意,主持人还举了如下的一个例子: 有一个数字串:312, 当N=3,K=1时会有以下两

[洛谷] P1018 乘积最大

题目描述 今年是国际数学联盟确定的“2000――世界数学年”,又恰逢我国著名数学家华罗庚先生诞辰90周年.在华罗庚先生的家乡江苏金坛,组织了一场别开生面的数学智力竞赛的活动,你的一个好朋友XZ也有幸得以参加.活动中,主持人给所有参加活动的选手出了这样一道题目: 设有一个长度为N的数字串,要求选手使用K个乘号将它分成K+1个部分,找出一种分法,使得这K+1个部分的乘积能够为最大. 同时,为了帮助选手能够正确理解题意,主持人还举了如下的一个例子: 有一个数字串:312, 当N=3,K=1时会有以下两

乘积最大洛谷p1018

题目描述 今年是国际数学联盟确定的"2000――世界数学年",又恰逢我国著名数学家华罗庚先生诞辰90周年.在华罗庚先生的家乡江苏金坛,组织了一场别开生面的数学智力竞赛的活动,你的一个好朋友XZ也有幸得以参加.活动中,主持人给所有参加活动的选手出了这样一道题目: 设有一个长度为N的数字串,要求选手使用K个乘号将它分成K+1个部分,找出一种分法,使得这K+1个部分的乘积能够为最大. 同时,为了帮助选手能够正确理解题意,主持人还举了如下的一个例子: 有一个数字串:312, 当N=3,K=1时

AC日记——乘积最大 洛谷 P1018

题目描述 今年是国际数学联盟确定的“2000――世界数学年”,又恰逢我国著名数学家华罗庚先生诞辰90周年.在华罗庚先生的家乡江苏金坛,组织了一场别开生面的数学智力竞赛的活动,你的一个好朋友XZ也有幸得以参加.活动中,主持人给所有参加活动的选手出了这样一道题目: 设有一个长度为N的数字串,要求选手使用K个乘号将它分成K+1个部分,找出一种分法,使得这K+1个部分的乘积能够为最大. 同时,为了帮助选手能够正确理解题意,主持人还举了如下的一个例子: 有一个数字串:312, 当N=3,K=1时会有以下两

洛谷 1018 乘积最大

题目描述 今年是国际数学联盟确定的“2000――世界数学年”,又恰逢我国著名数学家华罗庚先生诞辰90周年.在华罗庚先生的家乡江苏金坛,组织了一场别开生面的数学智力竞赛的活动,你的一个好朋友XZ也有幸得以参加.活动中,主持人给所有参加活动的选手出了这样一道题目: 设有一个长度为N的数字串,要求选手使用K个乘号将它分成K+1个部分,找出一种分法,使得这K+1个部分的乘积能够为最大. 同时,为了帮助选手能够正确理解题意,主持人还举了如下的一个例子: 有一个数字串:312, 当N=3,K=1时会有以下两

NYOJ 746 - 正整数n划分为m段,求m段的最大乘积 【区间DP】

整数划分(四) 时间限制:1000 ms  |  内存限制:65535 KB 描述 给出两个整数 n , m ,要求在 n 中加入m - 1 个乘号,将n分成m段,求出这m段的最大乘积 输入 第一行是一个整数T,表示有T组测试数据接下来T行,每行有两个正整数 n,m ( 1<= n < 10^19, 0 < m <= n的位数): 输出 输出每组测试样例结果为一个整数占一行 样例输入 2 111 2 1111 2 样例输出 11 121 #include <cstdio>