• 洛谷千题详解 | P1018 [NOIP2000 提高组] 乘积最大【C++、Python、Java、pascal语言】


    博主主页:Yu·仙笙

    专栏地址:洛谷千题详解

    目录

    题目描述

    输入格式

    输出格式

    输入输出样例

    解析:

    C++源码:

    Python源码:

    Pascal源码:

    Java源码:


    -------------------------------------------------------------------------------------------------------------------------------- 

    -------------------------------------------------------------------------------------------------------------------------------- 

    题目描述

    今年是国际数学联盟确定的“ 2000 ――世界数学年”,又恰逢我国著名数学家华罗庚先生诞辰 90 周年。在华罗庚先生的家乡江苏金坛,组织了一场别开生面的数学智力竞赛的活动,你的一个好朋友 XZ 也有幸得以参加。活动中,主持人给所有参加活动的选手出了这样一道题目:

    设有一个长度为 N 的数字串,要求选手使用 K 个乘号将它分成 K+1个部分,找出一种分法,使得这 K+1 个部分的乘积能够为最大。

    同时,为了帮助选手能够正确理解题意,主持人还举了如下的一个例子:

    有一个数字串:312, 当 N=3,K=1 时会有以下两种分法:

    1. 3×12=36
    2. 31×2=62

    这时,符合题目要求的结果是: 31×2=62

    现在,请你帮助你的好朋友 XZ 设计一个程序,求得正确的答案。

    -------------------------------------------------------------------------------------------------------------------------------- 

    输入格式

    程序的输入共有两行:

    第一行共有 2 个自然数 N,K(6≤N≤40,1≤K≤6)

    第二行是一个长度为 N 的数字串。

    -------------------------------------------------------------------------------------------------------------------------------- 

    输出格式

    结果显示在屏幕上,相对于输入,应输出所求得的最大乘积(一个自然数)。

    -------------------------------------------------------------------------------------------------------------------------------- 

    输入输出样例

    输入 #1

    4  2
    1231
    

    输出 #1

    62

    -------------------------------------------------------------------------------------------------------------------------------- 

    解析:

    用一个数组cut[i][j]存储在第i个数字后放第j个乘号,第1到第i个数的乘积的最大值。

    如果j=k,说明所有的乘号都已经放完,那么ans[i]就表示最后一个乘号放在第i个数后面的最大值,此时要乘上后面的数。因为后面的数是一定的,cut[i][j]是已知的最大值,所以ans[i]可以由唯一的路径转移。

    最后比较所有的ans[i],选择最大值输出。

    完成以上步骤需要至少三个操作:

    1.取数 将没有乘号分隔的连续的数字变成一个数,进行运算

    2.比较 没有比较哪来的最大值

    3.乘法 将乘号两边取到的数乘起来

    由于n<=40,所以这些操作要用高精度的方式进行

    (如果有能存40位的数据类型,请不必往下翻了,本蒟蒻最多知道一个long long)

    首先这道题很明显要dp做,因为这道题最后要求输出最大值,而该值可由两个上一级数字相乘得到。所以我们把原数先分为两块,一个由k段相乘得出的最大值和剩余部分的值,如图:

    而几段数的最大值则可以再次分割为右侧的一段数和左侧的几段数乘积的最大值。如此不断分割,直到分割到只剩一段,此时该段的最大值就是它本身。所以只要不断分割,每次分割都取当前情况的最优解即可得到问题最优解。

    接下来我们考虑怎么实现

    我们用a保存原数,a[i][j]表示原数的从第i个数到第j个数大小。

    举例:原数 124578

    则a[2][4]保存的值则为2457。(高精度下可以用三位数组,最后一位将每一位数分别储存)

    接下来我们用dp保存一段数的最大值,定义数组dp[][],dp[i][j]表示将前i个数分成j段可以得到的最大值。我们知道当该段获得最大值时,该段与该段后数字的乘积才能得到最大值。

    由此我们可以得到方程:

    *dp[i][j]=max(dp[i][j],dp[k-1][j-1]a[k][i])

    (高精度可以通过再数组上再扩充一维来保存每一位数)

    设字符串长度为n,乘号数为k,如果n=50,k=1时,

    有(n-1)=49种不同的乘法,当k=2时,有C(2,50-1)=1176种乘法,既C(k,n-1)种乘法,当n、k稍微大一些的时候,用穷举的方法就不行了。

    设数字字符串为a1a2…an

    K=1时:一个乘号可以插在a1a2…an中的n-1个位置,这样就得到n-1个子串的乘积:

    a1*a2…an, a1a2*a3…an, …, a1a2…a n-1*an (这相当于是穷举的方法)

    此时的最大值=max{a1*a2…an, a1a2*a3…an, … , a1a2…a n-1*an }

    K=2时,二个乘号可以插在a1a2…an中n-1个位置的任两个地方, 把这些乘积

    分个类,便于观察规律:

    ①倒数第一个数作为被乘数:

    a1*a2 …a n-3 a n-2 a n-1*an,

    a1a2 …*a n-2 a n-1*an,

    a1a2 …*a n-1*an。

    设符号F[n-1,1]为在前n-1个数中插入一个乘号的最大值,则的最大值为

    F[n-1,1]*an。

    ②倒数第二个数作为被乘数:

    a1*a2 …an-3 a n-2* a n-1,

    an … a1a2 …*a n-2*a n-1an,

    a1a2…*a n-3 a n-2* a n-1 an。

    设符号F[n-2,1]为在前n-2个数中插入一个乘号的最大值,则的最大值为

    F[n-2,1]*a n-1 an

    ③倒数第三个数作为被乘数:

    … 设符号F[n-3,1]为在前n-3个数中插入一个乘号的最大值,则的最大值为

    F[n-3,1]*a n-2 a n-1 an

    …… a3~an作为被乘数:F[2,1]*a3 …a n-2 a n-1 an

    此时的最大乘积为:

    F[n,k]=max{F[n-1]*an,F[n-2,1]*a n-1 an,

    F[n-3,1]*a n-2 a n-1 an,

    F[2,1]*a3 …a n-2 a n-1 an}

    设F表示在 i 个数中插入 j 个乘号的最大值,g表示从ai到aj的数字列,则可得到动态转移方程:

    F = max{F*g, F*g,

    F*g, …., F[j,j-1]*g[j+1,i]}

    (1<=i<=n, 1<=j<=k)

    边界: F =g[1,i] (数列本身)

    阶段:子问题是在子串中插入j-1,j-2……1,0个乘号,因此乘号个数作为阶段的划分(j个阶段)

    状态:每个阶段随着被乘数数列的变化划分状态。

    决策:在每个阶段的每种状态中做出决策。

    数据结构:

    一道DP老题。

    然而本菜鸡还是调了一下午+一晚上

    一开始想到记忆化搜索,也就是区间DP,枚举断点(乘号在的位置),然后瞎搞搞。

    后来发现可以不用记忆化框架

    令f[i][j]表示前i个数放j个乘号得到的最大值

    由于都是乘法,优先级一样,所以如果i之后还有乘号,可以之间调用这个f[i][j]进行运算,而无需考虑其内部的乘号是如何摆放的。

    所以状态转移的时候只需枚举最后一个乘号在的位置

    转移方程:f[i][j]=max(f[l][j-1]*num[l+1][i]);l是断点左边的位置,num[i][j],表示原序列i位到j为所组成的数。

    -------------------------------------------------------------------------------------------------------------------------------- 

    C++源码:

    1. #include
    2. #include
    3. using namespace std;
    4. int n,k,a[50];
    5. char s[50];
    6. struct node{//用结构体储存数组;当然,也可以直接用三维数组,不过感觉这样更容易理解
    7. int v;bool exi;//v:数位,exi:是否存在
    8. int c[50];//高精度数组
    9. }cut[50][10],ans[50];
    10. node culc(int l,int r){//取数操作,注意:要从右往左取,因为高精度数组是从低位往高位排的,而读入的数字串是从高位到低位
    11. node e;
    12. e.v=r-l+1;e.exi=true;
    13. for(int i=1;i<=e.v;i++){
    14. e.c[i]=a[r-i+1];
    15. }
    16. return e;
    17. }
    18. node mul(node e1,node e2){//高精度乘法
    19. node emul;
    20. emul.exi=true;emul.v=e1.v+e2.v-1;
    21. for(int i=1;i<=emul.v;i++) emul.c[i]=0;
    22. for(int i=1;i<=e1.v;i++)
    23. for(int j=1;j<=e2.v;j++)
    24. emul.c[i+j-1]+=e1.c[i]*e2.c[j];
    25. int q=0;
    26. for(int i=1;i<=emul.v;i++){
    27. emul.c[i]+=q;
    28. q=emul.c[i]/10;
    29. emul.c[i]%=10;
    30. }
    31. while(q>0){
    32. emul.c[++emul.v]=q%10;
    33. q/=10;
    34. }
    35. return emul;
    36. }
    37. node Max(node e1,node e2){//高精度比较,类似字符串(然而如果是字符串的话我就直接strcmp了)
    38. if(!e1.exi||e1.vreturn e2;
    39. if(!e2.exi||e2.vreturn e1;//先比较是否存在和位数
    40. for(int i=e1.v;i>=1;i--){//都存在,且位数相同,则逐位比较
    41. if(e1.c[i]>e2.c[i]) return e1;
    42. else if(e2.c[i]>e1.c[i]) return e2;
    43. }
    44. return e1;
    45. }
    46. int main(){
    47. scanf("%d%d",&n,&k);
    48. scanf("%s",s);
    49. for(int i=0;i1]=s[i]-'0';//将字符串变为数字数组
    50. for(int i=1;i<=n;i++){
    51. ans[i].exi=false;
    52. for(int j=1;j<=k;j++) cut[i][j].exi=false;
    53. }
    54. for(int i=1;i
    55. cut[i][1]=culc(1,i);//只放一个乘号的话不需要转移
    56. for(int j=2;j<=k;j++){
    57. for(int fr=j-1;fr//因为第i个数后放置的乘号最多是第i个,所以从j-1枚举front(前置位)
    58. if(cut[fr][j-1].exi) cut[i][j]=Max(cut[i][j],mul(cut[fr][j-1],culc(fr+1,i)));
    59. }
    60. } //转移状态
    61. if(cut[i][k].exi){
    62. ans[i]=mul(cut[i][k],culc(i+1,n));
    63. }
    64. }
    65. node lastans;lastans.exi=false;
    66. for(int i=1;i
    67. node tmp=Max(ans[i],lastans);
    68. lastans=tmp;//不知道为什么,直接写lastans=Max(lastans,ans[i])总是会错,然而加一个中间变量就过了。。
    69. }
    70. for(int i=lastans.v;i>=1;i--) printf("%d",lastans.c[i]);//输出
    71. return 0;
    72. }//写完注释感觉就像白痴代码一样啊。。。(内心:那你还写了半个上午???)

    -------------------------------------------------------------------------------------------------------------------------------- 

    Python源码

    1. n, k = map(int, input().split())
    2. s = int('1' + input())
    3. f = [[0 for i in range(k + 1)] for j in range(n + 1)]
    4. for i in range(1, n + 1):
    5. f[i][0] = int(str(s)[1: i + 1])
    6. for k1 in range(1, k + 1):
    7. for i in range(k1 + 1, n + 1):
    8. for j in range(k1, i):
    9. f[i][k1] = max(f[i][k1], f[j][k1 - 1] * int(str(s)[j + 1:i + 1]))
    10. print(f[n][k])

    -------------------------------------------------------------------------------------------------------------------------------- 

    Pascal源码:

    1. var
    2. n,k,i,j,i1,i2,i3:longint;
    3. f,sum:array[0..100,0..100] of ansistring;
    4. s:string;
    5. function gjc(a1,b1:string):string;
    6. var
    7. lena,lenb,lenc:longint;
    8. i,j,x:longint;
    9. a,b,c:array[0..200] of longint;
    10. k:string;
    11. begin
    12. fillchar(a,sizeof(a),0);
    13. fillchar(b,sizeof(b),0); //重置数组
    14. fillchar(c,sizeof(c),0);
    15. lena:=length(a1); lenb:=length(b1);
    16. for i:=1 to lena do a[lena-i+1]:=ord(a1[i])-48; //转化为数字
    17. for i:=1 to lenb do b[lenb-i+1]:=ord(b1[i])-48;
    18. for i:=1 to lena do
    19. begin
    20. x:=0; //高精度运算
    21. for j:=1 to lenb do
    22. begin
    23. c[i+j-1]:=a[i]*b[j]+x+c[i+j-1];
    24. x:=c[i+j-1] div 10;
    25. c[i+j-1]:=c[i+j-1] mod 10;
    26. end;
    27. c[i+j]:=x;
    28. end;
    29. lenc:=i+j;k:='';
    30. while (c[lenc]=0) and (lenc>1) do dec(lenc);
    31. for i:=lenc downto 1 do k:=k+chr(c[i]+48);
    32. exit(k); //返回字符串
    33. end;
    34. function max(a,b:string):string; //比较字符串的大小
    35. var
    36. lena,lenb:longint;
    37. begin
    38. lena:=length(a); lenb:=length(b);
    39. if (lena>lenb) then exit(a);
    40. if (lena
    41. if (lena=lenb) then
    42. if (a>b) then exit(a)
    43. else exit(b);
    44. end;
    45. begin
    46. readln(n,k);
    47. readln(s);
    48. for i:=1 to n do
    49. for j:=1 to n do
    50. sum[i,j]:=copy(s,i,j-i+1); //直接拷贝,当字符串处理
    51. for i:=1 to n do f[i,0]:=sum[1,i]; //初始化
    52. for i1:=1 to k do
    53. for i2:=i1+1 to n do
    54. for i3:=i1 to i2-1 do
    55. f[i2,i1]:=max(f[i2,i1],gjc(f[i3,i1-1],sum[i3+1,i2])); //高精计算返回字符串比较
    56. writeln(f[n,k]);
    57. end.

    -------------------------------------------------------------------------------------------------------------------------------- 

    Java源码:

    1. import java.io.BufferedReader;
    2. import java.io.IOException;
    3. import java.io.InputStream;
    4. import java.io.InputStreamReader;
    5. import java.io.OutputStream;
    6. import java.io.PrintWriter;
    7. import java.math.BigInteger;
    8. import java.util.StringTokenizer;
    9. public class Main {
    10. public static void main(String[] args) {
    11. // long sta = System.nanoTime();
    12. InputStream is = System.in;
    13. OutputStream os = System.out;
    14. IN cin = new IN(is);
    15. PrintWriter cout = new PrintWriter(os);
    16. SO so = new SO();
    17. so.solution(cin, cout);
    18. // long end = System.nanoTime();
    19. // cout.println("耗时:" + (double)(end-sta)/1e6 + "ms");
    20. cout.close();
    21. }
    22. static final int MOD = (int)1e9 + 7;
    23. static class SO {
    24. void solution(IN cin, PrintWriter cout) {
    25. int N = cin.nextInt(), K = cin.nextInt();
    26. BigInteger[][] num = new BigInteger[N][N];
    27. BigInteger[][] dp = new BigInteger[N][K+1];
    28. //初始化num数组
    29. char[] cs = cin.next().toCharArray();
    30. for(int i=0;i
    31. num[i][i] = BigInteger.valueOf(cs[i]-'0');
    32. for(int j=i+1;j
    33. num[i][j] = num[i][j-1].multiply(BigInteger.TEN).add(BigInteger.valueOf(cs[j]-'0'));
    34. }
    35. }
    36. //计算dp
    37. for(int i=0;i
    38. dp[i][0] = num[0][i];
    39. for(int i=0;i
    40. for(int k=1;k<=(i
    41. dp[i][k] = BigInteger.ZERO;
    42. for(int j=k-1;j
    43. BigInteger one = dp[j][k-1].multiply(num[j+1][i]);
    44. dp[i][k] = one.compareTo(dp[i][k])>0?one:dp[i][k];
    45. }
    46. }
    47. }
    48. cout.println(dp[N-1][K]);
    49. }//end solution
    50. }//end SO
    51. //以下是快读部分
    52. static class IN {
    53. private BufferedReader reader;
    54. private StringTokenizer tokenizer;
    55. IN(InputStream is) {
    56. reader = new BufferedReader(new InputStreamReader(is), 32768);
    57. tokenizer = null;
    58. }
    59. public String next() {
    60. while (tokenizer == null || !tokenizer.hasMoreTokens()) {
    61. try {
    62. tokenizer = new StringTokenizer(reader.readLine());
    63. } catch (IOException e) {
    64. throw new RuntimeException(e);
    65. }
    66. }
    67. return tokenizer.nextToken();
    68. }
    69. public int nextInt() {
    70. return Integer.parseInt(next());
    71. }
    72. public long nextLong() {
    73. return Long.parseLong(next());
    74. }
    75. public double nextDouble() {
    76. return Double.parseDouble(next());
    77. }
    78. }
    79. }

  • 相关阅读:
    六大赛题,108万丰厚奖池!2023长三角(芜湖)人工智能视觉算法大赛等你来战!
    Oracle 手工建库
    初识Java 11-1 函数式编程
    VC++OpenCV配置
    测试用例的设计
    windows远程连接服务器并映射端口访问目标服务
    IntelliJ IDEA 记学习笔 Maven自动导包 Auto Import
    2022-2028全球视频监控软件行业调研及趋势分析报告
    Linux 内存性能指标
    jenkins配置用户权限分配角色和视图(三)
  • 原文地址:https://blog.csdn.net/djfihhfs/article/details/127788541