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

news/2024/5/15 22:11:40/文章来源:https://blog.csdn.net/djfihhfs/article/details/127788541

博主主页: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++源码:

#include<cstdio>
#include<cstring>
using namespace std;
int n,k,a[50];
char s[50];
struct node{//用结构体储存数组;当然,也可以直接用三维数组,不过感觉这样更容易理解int v;bool exi;//v:数位,exi:是否存在int c[50];//高精度数组
}cut[50][10],ans[50];
node culc(int l,int r){//取数操作,注意:要从右往左取,因为高精度数组是从低位往高位排的,而读入的数字串是从高位到低位node e;e.v=r-l+1;e.exi=true;for(int i=1;i<=e.v;i++){e.c[i]=a[r-i+1];}    return e;
}
node mul(node e1,node e2){//高精度乘法node emul;emul.exi=true;emul.v=e1.v+e2.v-1;for(int i=1;i<=emul.v;i++) emul.c[i]=0;for(int i=1;i<=e1.v;i++)for(int j=1;j<=e2.v;j++)emul.c[i+j-1]+=e1.c[i]*e2.c[j];int q=0;        for(int i=1;i<=emul.v;i++){emul.c[i]+=q;q=emul.c[i]/10;emul.c[i]%=10;}while(q>0){emul.c[++emul.v]=q%10;q/=10;}return emul;
}
node Max(node e1,node e2){//高精度比较,类似字符串(然而如果是字符串的话我就直接strcmp了)if(!e1.exi||e1.v<e2.v) return e2;if(!e2.exi||e2.v<e1.v) return e1;//先比较是否存在和位数for(int i=e1.v;i>=1;i--){//都存在,且位数相同,则逐位比较if(e1.c[i]>e2.c[i]) return e1;else if(e2.c[i]>e1.c[i]) return e2;}return e1;
}
int main(){scanf("%d%d",&n,&k);scanf("%s",s);for(int i=0;i<n;i++) a[i+1]=s[i]-'0';//将字符串变为数字数组for(int i=1;i<=n;i++){ans[i].exi=false;for(int j=1;j<=k;j++) cut[i][j].exi=false;}for(int i=1;i<n;i++){cut[i][1]=culc(1,i);//只放一个乘号的话不需要转移for(int j=2;j<=k;j++){for(int fr=j-1;fr<i;fr++){//因为第i个数后放置的乘号最多是第i个,所以从j-1枚举front(前置位)if(cut[fr][j-1].exi) cut[i][j]=Max(cut[i][j],mul(cut[fr][j-1],culc(fr+1,i)));}} //转移状态if(cut[i][k].exi){ans[i]=mul(cut[i][k],culc(i+1,n));}}node lastans;lastans.exi=false;for(int i=1;i<n;i++){node tmp=Max(ans[i],lastans);lastans=tmp;//不知道为什么,直接写lastans=Max(lastans,ans[i])总是会错,然而加一个中间变量就过了。。}for(int i=lastans.v;i>=1;i--) printf("%d",lastans.c[i]);//输出return 0;
}//写完注释感觉就像白痴代码一样啊。。。(内心:那你还写了半个上午???)

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

Python源码:

n, k = map(int, input().split())
s = int('1' + input())
f = [[0 for i in range(k + 1)] for j in range(n + 1)]
for i in range(1, n + 1):f[i][0] = int(str(s)[1: i + 1])
for k1 in range(1, k + 1):for i in range(k1 + 1, n + 1):for j in range(k1, i):f[i][k1] = max(f[i][k1], f[j][k1 - 1] * int(str(s)[j + 1:i + 1]))
print(f[n][k])

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

Pascal源码:

varn,k,i,j,i1,i2,i3:longint;f,sum:array[0..100,0..100] of ansistring;s:string;function gjc(a1,b1:string):string;
varlena,lenb,lenc:longint;i,j,x:longint;a,b,c:array[0..200] of longint;k:string;
beginfillchar(a,sizeof(a),0);  fillchar(b,sizeof(b),0);  //重置数组fillchar(c,sizeof(c),0);lena:=length(a1); lenb:=length(b1);for i:=1 to lena do a[lena-i+1]:=ord(a1[i])-48;  //转化为数字for i:=1 to lenb do b[lenb-i+1]:=ord(b1[i])-48;for i:=1 to lena do  begin x:=0; 	            //高精度运算                                                 for j:=1 to lenb dobegin                                c[i+j-1]:=a[i]*b[j]+x+c[i+j-1];              x:=c[i+j-1] div 10;c[i+j-1]:=c[i+j-1] mod 10;end;c[i+j]:=x;                                               end;lenc:=i+j;k:='';while (c[lenc]=0) and (lenc>1) do dec(lenc);for i:=lenc downto 1 do k:=k+chr(c[i]+48);exit(k);  //返回字符串
end;function max(a,b:string):string;  //比较字符串的大小
varlena,lenb:longint;
beginlena:=length(a); lenb:=length(b);if (lena>lenb) then exit(a);if (lena<lenb) then exit(b);if (lena=lenb) then if (a>b) then exit(a)else exit(b);
end;beginreadln(n,k);readln(s);for i:=1 to n dofor j:=1 to n dosum[i,j]:=copy(s,i,j-i+1);  //直接拷贝,当字符串处理for i:=1 to n do f[i,0]:=sum[1,i]; //初始化for i1:=1 to k dofor i2:=i1+1 to n dofor i3:=i1 to i2-1 dof[i2,i1]:=max(f[i2,i1],gjc(f[i3,i1-1],sum[i3+1,i2]));  //高精计算返回字符串比较writeln(f[n,k]);
end.

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

Java源码:

import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStream;
import java.io.InputStreamReader;
import java.io.OutputStream;
import java.io.PrintWriter;
import java.math.BigInteger;
import java.util.StringTokenizer;public class Main {public static void main(String[] args) {
//		long sta = System.nanoTime();InputStream is = System.in;OutputStream os = System.out;IN cin = new IN(is);PrintWriter cout = new PrintWriter(os);SO so = new SO();so.solution(cin, cout);//		long end = System.nanoTime();
//		cout.println("耗时:" + (double)(end-sta)/1e6 + "ms");cout.close();}static final int MOD = (int)1e9 + 7;static class SO {void solution(IN cin, PrintWriter cout) {int N = cin.nextInt(), K = cin.nextInt();BigInteger[][] num = new BigInteger[N][N];BigInteger[][] dp = new BigInteger[N][K+1];//初始化num数组char[] cs = cin.next().toCharArray();for(int i=0;i<N;++i) {num[i][i] = BigInteger.valueOf(cs[i]-'0');for(int j=i+1;j<N;++j) {num[i][j] = num[i][j-1].multiply(BigInteger.TEN).add(BigInteger.valueOf(cs[j]-'0'));}}//计算dpfor(int i=0;i<N;++i)dp[i][0] = num[0][i];for(int i=0;i<N;++i) {for(int k=1;k<=(i<K?i:K);++k) {dp[i][k] = BigInteger.ZERO;for(int j=k-1;j<i;++j) {BigInteger one = dp[j][k-1].multiply(num[j+1][i]);dp[i][k] = one.compareTo(dp[i][k])>0?one:dp[i][k];}}}cout.println(dp[N-1][K]);}//end solution}//end SO//以下是快读部分static class IN {private BufferedReader reader;private StringTokenizer tokenizer;IN(InputStream is) {reader = new BufferedReader(new InputStreamReader(is), 32768);tokenizer = null;}public String next() {while (tokenizer == null || !tokenizer.hasMoreTokens()) {try {tokenizer = new StringTokenizer(reader.readLine());} catch (IOException e) {throw new RuntimeException(e);}}return tokenizer.nextToken();}public int nextInt() {return Integer.parseInt(next());}public long nextLong() {return Long.parseLong(next());}public double nextDouble() {return Double.parseDouble(next());}}
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.luyixian.cn/news_show_223818.aspx

如若内容造成侵权/违法违规/事实不符,请联系dt猫网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

苯丙氨酸甲酯双三氟甲基磺酰亚胺[PheC1][Tf2N]氨基酸酯离子液体

苯丙氨酸甲酯双三氟甲基磺酰亚胺[PheC1][Tf2N]氨基酸酯离子液体 纯度&#xff1a;95% 外观与形状:液体/固体, 储存:存放于惰性气体之中 应避免湿气 (吸湿) 包装规格(Packing):50g、100g、500g 保存方法&#xff1a;密闭&#xff0c;阴凉&#xff0c;通风干燥处 氨基酸酯…

返回Series或DataFrame中指定列中指定数量的最小值nsmallest()函数

【小白从小学Python、C、Java】 【计算机等级考试500强双证书】 【Python-数据分析】 返回Series或DataFrame中 指定列中指定数量的最小值 nsmallest()函数 [太阳]选择题 下列说法错误的是? import pandas as pd mySeries pd.Series([31, 21, 11]) print("【显示】mySer…

Numpy手撸softmax regression

算法介绍 Softmax 回归&#xff08;或多项逻辑回归&#xff09;是将逻辑回归推广到我们想要处理多个类的情况。 在逻辑回归中&#xff0c;我们假设标签是二元的&#xff1a;y(i)∈{0,1}y^{(i)} \in \{0,1\}y(i)∈{0,1},我们使用这样的分类器来区分两种手写数字。 Softmax 回归…

C#项目实战|人脸识别考勤

此文主要通过WinForm来制作的一个人脸识别考勤打卡程序&#xff0c;有兴趣的小伙伴可以接入到打卡机上。 一、实现流程1.1、创建项目1.2、设计页面1.3、创建应用1.4、获取Token及参数解析1.5、与人脸数据比对并展示一、实现流程 1.1、创建项目 打开Visual Studio&#xff0c;右…

值得入手的键盘——Keychron K8 Pro

目录 一、前言 二、介绍 三、上手体验 四、总结 一、前言 在如今&#xff0c;外设产品市场相当火爆的时代&#xff0c;拥有诸多知名的品 牌&#xff0c;而一个新品牌要在竞争非常激烈的情况下站稳脚跟&#xff0c;实属不易。诞生于2017年的 Keychron 以其品质作为高端战略…

【mcuclub】舵机-SG90

一、实物图&#xff08;SG90&#xff09; 二、原理图 编号名称功能1GND电源地&#xff08;棕色线&#xff09;2VCC电源正&#xff08;红色线&#xff09;3I/O信号线&#xff08;黄色线&#xff09; 三、简介 舵机&#xff08;英文叫Servo&#xff09;&#xff0c;是伺服电机的…

WINDOWS核心编程--Windows程序内部运行机制

现代的桌面应用基本上很少使用原始的 Windows API 进行开发了&#xff0c;因为使用原始 API 堆砌出来的应用代码逻辑非常繁琐&#xff0c;特别是窗口消息的处理非常不方便&#xff0c;大多数直接使用 C# 或者 QT 这种跨平台的开发库&#xff0c;而那种直接封装 Windows API 而存…

C语言经典题目之青蛙跳台阶问题

目录 一、问题描述 二、问题分析 1.当n1时 2.当n2时 3.当n3时 4.n4&#xff0c;n5........nn时 三、代码实现 总结 一、问题描述 一只青蛙一次可以跳上 1 级台阶&#xff0c;也可以跳上2 级。求该青蛙跳上一个n 级的台阶总共有多少种跳法。 二、问题分析 青蛙跳台阶&a…

Spring-Aop面向切面编程

文章目录一、简介1、作用2、AOP核心概念3、五种&#xff08;增强&#xff09;通知类型二、AOP入门小案例&#xff08;注解版&#xff09;1.导入坐标(pom.xml)2.制作连接点(原始操作&#xff0c;Dao接口与实现类)3:定义通知类和通知4:定义切入点5:制作切面6:将通知类配给容器并标…

【Linux】第十一章 进程信号(概念+产生信号+阻塞信号+捕捉信号)

&#x1f3c6;个人主页&#xff1a;企鹅不叫的博客 ​ &#x1f308;专栏 C语言初阶和进阶C项目Leetcode刷题初阶数据结构与算法C初阶和进阶《深入理解计算机操作系统》《高质量C/C编程》Linux ⭐️ 博主码云gitee链接&#xff1a;代码仓库地址 ⚡若有帮助可以【关注点赞收藏】…

C++基本知识(二)---函数重载、引用、内联函数、auto关键字

目录 1.函数重载 2.引用 3.内联函数 4.auto关键字(C11) 5.指针空值nullptr(C11) 1.函数重载 重载函数是函数的一种特殊情况&#xff0c;为方便使用&#xff0c;C允许在同一范围中声明几个功能类似的同名函数&#xff0c;但是这些同名函数的形式参数&#xff08;指参数的个…

CEC2015:(二)动态多目标野狗优化算法DMODOA求解DIMP2、dMOP2、dMOP2iso、dMOP2dec(提供Matlab代码)

一、cec2015中测试函数DIMP2、dMOP2、dMOP2iso、dMOP2dec详细信息 CEC2015&#xff1a;动态多目标测试函数之DIMP2、dMOP2、dMOP2iso、dMOP2dec详细信息 二、动态多目标野狗优化算法 多目标野狗优化算法&#xff08;Multi-Objective Dingo Optimization Algorithm&#xff0…

瑞吉外卖强化(一):缓存优化

瑞吉外卖强化&#xff08;一&#xff09;&#xff1a;缓存优化瑞吉外卖 缓存优化Redis基本操作短信验证码 缓存实现缓存菜品数据SpringCache常用注解瑞吉外卖 缓存优化 Redis基本操作 redisTemplate需要配置类 这里的 需要对其进行 序列化操作 reidsTeplate.opsForValue().s…

论文精读:Swin Transformer V2: Scaling Up Capacity and Resolution

论文地址:https://arxiv.org/pdf/2111.09883.pdf 代码地址: GitHub - microsoft/Swin-Transformer: This is an official implementation for "Swin Transformer: Hierarchical Vision Transformer using Shifted Windows". Abstract 本篇论文主要致力于解决大型…

TCP三次握手和四次挥手基本知识

一、概述 TCP是面向连接、可靠的、基于字节流的传输层通讯协议。 如何确定一个TCP连接&#xff1a; 目的IP目的端口源IP源端口 二、TCP建立连接 序列号client_isn和server_isn是随机初始化&#xff0c;可以通过netstat -napt来查看网络状态。 为什么建立连接需要三次握手&…

c++哈希(哈希表闭散列线性探测实现)

文章目录0. 前言1. 线性探测2. 线性探测的代码实现2.0 定义2.1 插入实现--Insert2.2 查找实现--Find2.3 删除实现--Erase2.4 仿函数3. 完整代码实现4. 代码测试并运行结果&#xff1a;0. 前言 闭散列&#xff1a;也叫开放定址法&#xff0c;当发生哈希冲突时&#xff0c;如果哈…

Python画爱心——谁能拒绝用代码敲出来会跳动的爱心呢~

还不快把这份浪漫拿走&#xff01;&#xff01;节日就快到来了&#xff0c;给Ta一个惊喜吧~ 今天给大家分享一个浪漫小技巧&#xff0c;利用Python制作一个立体会动的心动小爱心 成千上百个爱心汇成一个大爱心&#xff0c;从里到外形成一个立体状&#xff0c;给人视觉上的冲击…

年轻人不用太过于努力

周末和一个毕业一年多的朋友聊天&#xff0c;我随口问了一句「你有什么想跟我分享的」&#xff0c;然后他就说了上面的那句话。「年轻人不用太过于努力」和读者聊天会做成我的一个公众号专栏&#xff0c;内容有也会越来越丰富&#xff0c;全部的内容都会收录到我的程序人生专栏…

采购管理主要流程有哪些?

采购管理流程是很多企业用于获取物资或服务的一种关键步骤。采购管理流程对企业至关重要&#xff0c;因为它们可以对利润和支出产生会有直接的影响。 由于各个企业有不同的需求和目标&#xff0c;采购管理流程可能会有所不同。虽然与其采购流程相关的细节可能有所不同&#xf…

web前端课程设计——动漫网页2个网页HTML+CSS web前端开发技术 web课程设计 网页规划与设计

HTML实例网页代码, 本实例适合于初学HTML的同学。该实例里面有设置了css的样式设置&#xff0c;有div的样式格局&#xff0c;这个实例比较全面&#xff0c;有助于同学的学习,本文将介绍如何通过从头开始设计个人网站并将其转换为代码的过程来实践设计。 ⚽精彩专栏推荐&#x1…