最大 k 乘积问题 ( 经典区间DP )
题意 : 设 NUM 是一个 n 位十进制整数。如果将 NUM 划分为 k 段,则可得到 k 个整数。这 k 个整数的乘积称为 NUM 的一个 k 乘积。试设计一个算法,对于给定的 NUM 和 k,求出 NUM 的最大 k 乘积
分析 :
定义 dp[i][j] = 前 i 个数字中间插入 j 个乘号时候的最大乘积是多少
初始化 dp[ i ][ 0 ] = NUM(1, i) 1 <= i <= len(NUM)
最后的结果则存于 dp[n][k]
状态转移方程为 dp[i][j] = max( dp[i][j] , dp[m][j-1] * NUM[m+1 ~ i] ) j-1 < m < i
注 : NUM[a~b] 表示 NUM 的第 a 位到第 b 位组成的数字
关于状态转移方程先来看一个例子 n = 4、k = 2、NUM = 1231
首先初始化
dp[1][0] = 1
dp[2][0] = 12
dp[3][0] = 123
dp[4][0] = 1231
然后安插 1 个乘号的时候各个长度的最大乘积
dp[2][1] = dp[1][0] * 2 = 2
dp[3][1] = max( dp[1][0]*23、dp[2][0]*3 ) = 36
dp[4][1] = max( dp[1][0]*231、dp[2][0]*31、dp[3][0]*1 ) = 372
接着是安插 2 个乘号的时候
dp[3][2] = dp[2][1] * 3 = 6
dp[4][2] = max( dp[2][1]*31、dp[3][1]*1 ) = 62
细细去推一下这个例子,可能就会发现更加理解了这个 dp
可以看出这个 dp 定义的第二维应该是阶段、而第一维是 dp 的状态
换句话说只有知道在各个长度安插 1 个乘号的结果才能推出各个长度安插 2 个乘号的结果
C++版
#include<bits/stdc++.h>
#define LL long long
using namespace std;
;
LL dp[maxn][maxn];
char num[maxn];
LL GetVal(int st, int en)
{
LL ret = ;
for(int i=st; i<=en; i++)
ret = ret * + (num[i] - ');
return ret;
}
int main(void)
{
int n, k;
while(~scanf("%d %d", &n, &k)){
scanf());
; i<=n; i++)
dp[i][] = GetVal(, i);
; j<=k-; j++)
; i<=n; i++)
for(int m=j; m<i; m++)
dp[i][j] = max(dp[i][j], dp[m][j-]*GetVal(m+, i));
printf(]);
}
;
}
JAVA大数版
import java.io.*;
import java.lang.reflect.Array;
import java.util.*;
import java.math.*;
import java.util.Arrays;
public class Main {
static BigInteger[][] dp = new BigInteger[15][15];
public static void main(String[] args){
Scanner cin = new Scanner (new BufferedInputStream(System.in));
int n, k;
String s;
while(cin.hasNext()){
n = cin.nextInt();
k = cin.nextInt();
s = cin.next();
for(int i=1; i<=n; i++){
dp[i][0] = BigInteger.valueOf( Integer.parseInt(s.substring(0,i)) );
//System.out.println(dp[i][0]);
}
for(int j=1; j<=k-1; j++)
for(int i=j+1; i<=n; i++){
dp[i][j] = BigInteger.ZERO;
for(int m=j; m<i; m++){
if(dp[i][j].compareTo(dp[m][j-1].multiply( BigInteger.valueOf( Integer.parseInt(s.substring(m, i)) ) )) < 0)
dp[i][j] = dp[m][j-1].multiply( BigInteger.valueOf( Integer.parseInt(s.substring(m, i)) ) );
}
}
System.out.println(dp[n][k-1]);
}
}
}
最大 k 乘积问题 ( 经典区间DP )的更多相关文章
- POJ 1160 经典区间dp/四边形优化
链接http://poj.org/problem?id=1160 很好的一个题,涉及到了以前老师说过的一个题目,可惜没往那上面想. 题意,给出N个城镇的地址,他们在一条直线上,现在要选择P个城镇建立邮 ...
- poj 1390 Blocks (经典区间dp 方块消除)
Blocks Time Limit: 5000MS Memory Limit: 65536K Total Submissions: 4250 Accepted: 1704 Descriptio ...
- 蓝桥杯 乘积最大(区间dp、数据水的话long long,暴力就能过)
Description 今年是国际数学联盟确定的“2000——世界数学年”,又恰逢我国著名数学家华罗庚先生诞辰90周年.在华罗庚先生的家乡江苏金坛,组织了一场别开生面的数学智力竞赛的活动,你的一个好朋 ...
- ZOJ 3541 The Last Puzzle(经典区间dp)
http://acm.zju.edu.cn/onlinejudge/showProblem.do?problemCode=3541 题意:有一排开关,有个开关有两个值t和d,t是按下开关后在t秒后会自 ...
- nyoj 737 石子合并 经典区间 dp
石子合并(一) 时间限制:1000 ms | 内存限制:65535 KB 难度:3 描述 有N堆石子排成一排,每堆石子有一定的数量.现要将N堆石子并成为一堆.合并的过程只能每次将相邻的两堆 ...
- 51nod 1021 石子归并 - 区间dp(经典)
题目地址:http://www.51nod.com/onlineJudge/questionCode.html#!problemId=1021 经典区间dp,dp[i][j] 表示将从 i 到 j 堆 ...
- 区间dp实战练习
题解报告:poj 2955 Brackets(括号匹配) Description We give the following inductive definition of a “regular br ...
- [NYIST15]括号匹配(二)(区间dp)
题目链接:http://acm.nyist.edu.cn/JudgeOnline/problem.php?pid=15 经典区间dp,首先枚举区间的大小和该区间的左边界,这时右边界也可计算出来.首先初 ...
- 题解报告:NYOJ #737 石子合并(一)(区间dp)
描述 有N堆石子排成一排,每堆石子有一定的数量.现要将N堆石子并成为一堆.合并的过程只能每次将相邻的两堆石子堆成一堆,每次合并花费的代价为这两堆石子的和,经过N-1次合并后成为一堆.求出总的代价最小值 ...
随机推荐
- python每日一练:0005题
第 0005 题: 你有一个目录,装了很多照片,把它们的尺寸变成都不大于 iPhone5 分辨率的大小. import cv2 import os def resize(path,sizeX,size ...
- Github 上 Star 最多的个人 Spring Boot 开源学习项目(三)
网上连载了 Spring Boot 系列文章 这个开源项目就是 spring-boot-examples ,这是一个专注帮助初学者学习 Spring Boot 的开源项目,里面分享了各种场景下 Spr ...
- JDK安装中配置Path无效解决办法
1. 问题 在安装jdk后,配置完环境变量,然后在控制台输入java -version出现与安装版本不一致的版本,如安装1.8后version仍显示1.7,即:修改环境变量没有生效且原先存在安装过的J ...
- Two modules in a project cannot share the same content root报错解决方案
观察上方是否出现两个同样的项目,删除不需要的那个,我觉得是因为两个项目同时引用一个根目录文件导致的.
- java 数组详细介绍
一.概述 数组(Array),是多个相同类型数据按一定顺序排列的集合,并使用一个名字命名,并通过编号的方式对这些数据进行统一管理 数组常见概念: 数组名, 下标(或索引), 元素, 数组的长度 数组本 ...
- PHP 经典有趣的算法
原文:https://blog.csdn.net/a519395243/article/details/77942913 1.一群猴子排成一圈,按1,2,…,n依次编号.然后从第1只开始数,数到第m只 ...
- Ubantu上安装Redis
Ubantu上安装Redis:Redis(Remote Dictionary Server):远程字典服务器,简称REDIS;Redis数据库产品用C语言编写而成,开源.少量数据存储.高速读写访问,是 ...
- Min-Max 容斥的证明
这里有 Min-Max 容斥的证明以及唯一一道博主做过的例题... 上个结论: \[Min\{S\}=\sum_{T\subseteq S,T\not=\varnothing}(-1)^{|T|-1} ...
- 通过编写串口助手工具学习MFC过程——(八)遇到的一些问题
通过编写串口助手工具学习MFC过程 因为以前也做过几次MFC的编程,每次都是项目完成时,MFC基本操作清楚了,但是过好长时间不再接触MFC的项目,再次做MFC的项目时,又要从头开始熟悉.这次通过做一个 ...
- PCIe基础篇(二)、协议详解
一个完整的PCIe协议体系结构包括应用层.事务层(Transaction Layer).数据链路层(Data Link Layer)和物理层(Physical Layer).其中,应用层由用户需要自行 ...