991 AlvinZH的奇幻猜想----整数乘积plus(背包DP大作战P)
914 AlvinZH的奇幻猜想----整数乘积puls
思路
难题。动态规划。
将数字串按字符串输入,处理起来更方便些。
dp[i][j]:表示str[0~i]中插入j个乘号时的乘积最大值。状态转移方程为:dp[i][j] = max(dp[i][j], dp[i-k][j-1]*convert(i-k+1,i)),k∈[1,i-j+1],convert为数字转换函数。
注意:本题在上一题的基础上,数字串变得很长,long long根本处理不了。
方法:大数乘法。定义大数结构体,用数组存下每一位数字,做乘法时一位一位处理。建议百度学习一下大数的加减乘除,这是常见的一个面试题。具体见参考代码一。
当然,你要是会java的话,直接用BigInteger也行,简单,时间会大了点,不影响过题。具体见参考代码二。
参考代码一
//
// Created by AlvinZH on 2017/10/31.
// Copyright (c) AlvinZH. All rights reserved.
//
#include <cstdio>
#include <cstring>
#include <iostream>
using namespace std;
struct BigNum {
int len;
int num[100];
};
int n, kk;
char s[105];//数字串
int A[105];//原数字
BigNum dp[105][12];//dp[i][j]表示前i个数字有j个乘号时的最大值。
BigNum cal(BigNum x, int l, int r)
{
BigNum ans, y;
memset(ans.num, 0, sizeof(ans.num));
memset(y.num, 0, sizeof(y.num));
y.len = r - l + 1;
for (int i = r; i >= l; --i)
y.num[r-i+1] = A[i];
for (int i = 1; i <= x.len; ++i) {
for (int j = 1; j <= y.len; ++j) {
ans.num[i+j-1] += x.num[i] * y.num[j];
}
}
int newLen = x.len + y.len -1;
for (int i = 1; i <= newLen; ++i) {
ans.num[i+1] += ans.num[i]/10;
ans.num[i] = ans.num[i]%10;
}
if(ans.num[newLen+1] > 0) newLen++;
ans.len = newLen;
/*for (int i = ans.len; i >= 1; --i)
printf("%d", ans.num[i]);
printf("\n");*/
return ans;
}
BigNum cmp(BigNum x, BigNum y)//比较大小
{
if(x.len > y.len) return x;
else if(x.len < y.len) return y;
for (int i = x.len; i >=1 ; --i) {
if(x.num[i] > y.num[i]) return x;
else if(x.num[i] < y.num[i]) return y;
}
return x;
}
int main()
{
//freopen("in1.txt", "r", stdin);
//freopen("outme.txt", "w", stdout);
while(~scanf("%d %s", &kk, s))
{
for (int i = 0; i < 105; ++i) {
for (int j = 0; j < 12; ++j) {
dp[i][j].len = 0;
memset(dp[i][j].num, 0, sizeof(dp[i][j]));
}
}
n = strlen(s);
for (int i = 1; i <= n; ++i)
A[i] = s[i-1] - '0';
//初始化没有添加乘号的情况
for (int i = 1; i <= n; ++i) {
for (int j = i; j >= 1; --j) {
dp[i][0].num[++dp[i][0].len] = A[j];
}
}
for (int i = 2; i <= n; ++i) {
for (int k = 1; k <= min(kk, i-1); ++k) {
for (int j = k; j < i; ++j) {
dp[i][k] = cmp(dp[i][k], cal(dp[j][k-1], j+1, i));
}
}
}
for (int i = dp[n][kk].len; i >= 1; --i) {
printf("%d", dp[n][kk].num[i]);
if(dp[n][kk].num[dp[n][kk].len] == 0) break;
}
printf("\n");
}
}
参考代码二
/*
Author: 赵立晨(12657)
Result: AC Submission_id: 402925
Created at: Sun Nov 12 2017 13:24:46 GMT+0800 (CST)
Problem: 914 Time: 451 Memory: 48880
*/
//package main;
import java.math.BigInteger;
import java.util.Scanner;
public class Main {
public static void main(String args[]){
BigInteger dp[][]=new BigInteger[107][17];
BigInteger num[][]=new BigInteger[107][107];
Scanner in=new Scanner(System.in);
while (in.hasNext()){
for (int i=0;i<=100;i++)
for (int j=0;j<=10;j++) dp[i][j]=BigInteger.ZERO;
int n=in.nextInt();
String a=in.next();
for (int i=0;i<a.length();i++){
for (int j=i;j<a.length();j++){
num[i][j]=new BigInteger(a.substring(i,j+1));
}
}
dp[0][0]=BigInteger.ZERO;
for (int i=0;i<a.length();i++) dp[i][0]=num[0][i];
for (int i=0;i<a.length();i++){
for (int k=1;k<=n;k++){
for (int j=1;j<=i;j++){
dp[i][k]=dp[i][k].max(dp[j-1][k-1].multiply(num[j][i]));
}
}
}
System.out.println(dp[a.length()-1][n]);
}
in.close();
}
}
991 AlvinZH的奇幻猜想----整数乘积plus(背包DP大作战P)的更多相关文章
- 906 AlvinZH的奇幻猜想----整数乘积(背包DP大作战O)
906 AlvinZH的奇幻猜想----整数乘积 思路 难题.动态规划. 将数字串按字符串输入,处理起来更方便些. dp[i][j]:表示str[0~i]中插入j个乘号时的乘积最大值.状态转移方程为: ...
- AlvinZH掉坑系列讲解(背包DP大作战H~M)
本文由AlvinZH所写,欢迎学习引用,如有错误或更优化方法,欢迎讨论,联系方式QQ:1329284394. 前言 动态规划(Dynamic Programming),是一个神奇的东西.DP只能意会, ...
- 977 AlvinZH过生日(背包DP大作战S)
977 AlvinZH过生日 思路 难题.逆推DP. 要明确dp的状态只与是否有选择权有关,而与选择权在谁手里无关.因为不论选择权在谁手里,那个人都会尽可能的获得最大的蛋糕重量. dp[i]表示分配到 ...
- 976 AlvinZH想回家(背包DP大作战T)
976 AlvinZH想回家 思路 如果在第i小时有一些飞机延误,那么一架飞机的c值越大,这一小时产生的损失也越大.而使这一小时产生的损失尽可能的小并不会导致接下来时间产生的损失增大.因此应当每一小时 ...
- 963 AlvinZH打怪刷经验(背包DP大作战R)
963 AlvinZH打怪刷经验 思路 这不是一道普通的01背包题.大家仔细观察数据的范围,可以发现如果按常理来的话,背包容量特别大,你也会TLE. 方法一:考虑01背包的一个常数优化----作用甚微 ...
- 851 AlvinZH的鬼畜密码(背包DP大作战N)
851 AlvinZH的鬼畜密码 思路 难题.动态规划. 先判断字符串是否合理(可翻译),然后分段处理,每一小段用动态规划求出解法数. dp[i]:字符串str[0~i]的解法数.通过判断str[i] ...
- 2016级算法第三次上机-C.AlvinZH的奇幻猜想——三次方
905 AlvinZH的奇幻猜想--三次方 思路 中等题.题意简单,题目说得简单,把一个数分成多个立方数的和,问最小立方数个数. 脑子转得快的马上想到贪心,从最近的三次方数往下减,反正有1^3在最后撑 ...
- 算法-找出与目标数字相同的digit组成的整数中比该数字大的数集中的最小数字
题目: 给出1个正整数,找到用与这个数字相同的digit组成的整数中比这个数字大的数集中的最小数字.比如:12352874 的结果是 12354278 分析: 这道题目的考虑目标是数组的查找与排序. ...
- Ural 1158. Censored! 有限状态自动机+DP+大整数
Ural1158 看上去很困难的一道题. 原文地址 http://blog.csdn.net/prolightsfxjh/article/details/54729646 题意:给出n个不同的字符,用 ...
随机推荐
- 关于防SQL注入敏感词过滤问题
关于对字符的过滤问题sql查询条件过滤掉单引号是否就安全了呢? 在文章最后一段管理员做了敏感字符的过滤,管理员过滤掉了空格,而攻击者通过 /**/ 来代替空格绕过了过滤字符.感觉很有成就感,呵呵呵呵. ...
- swiper 下拉刷新混乱
在下来刷新时,banner轮播图播放混乱: 添加判断 if(this.mySwiper.activeIndex){ this.mySwiper.destroy(true,false);//这句是关键, ...
- 打开程序出现.Net Framework Initialization Error – Unable to find a version of the runtime to run this applicatio的解决办法
部署一个VS2010开发的程序时遇到 了一个非常奇怪的问题,客户端上已经安装了.net framework 4.0,但运行时还是会弹出错误: .Net Framework Initialization ...
- java 集合综述(总结于多位博友)
http://www.cnblogs.com/shunran/p/3459065.html(good) java集合类主要负责保存.盛装其他数据,因此集合类也称容器类. java集合类分为:set.l ...
- hibernate 多表联合查询
以前用sql实现联合查询 是非常简单的事,只需要写sql语句就可以,第一次遇到hibernate要实现多表联合查询的时候还楞了一下.最后看了下资料,才恍然大悟,hibernate实现多表联合查询跟SQ ...
- javascript的那些事儿你都懂了吗
javascript从开始的验证表单的脚本语言发展到现在能运行在服务器上,其影响力不断的提升.自己作为一个做前端的,编写js是必不可少,从自己学习js的历程来看其实也是比较吃力.要 学好它,还是的花费 ...
- javascript的caller,callee,call,apply[转]
在提到上述的概念之前,首先想说说javascript中函数的隐含参数:arguments Arguments 该对象代表正在执行的函数和调用它的函数的参数. [function.]arguments[ ...
- 6、Semantic-UI之动画按钮样式
6.1 动画按钮样式 在Semantic-UI中提供了三种动画样按钮式表,分别为: 左右移动 上下移动 淡入淡出 在实际开发中,很少使用这种动画按钮,根据实际情况使用,强制使用到页面中反而不太适合 ...
- Reporting Service服务SharePoint集成模式安装配置(3、4、安装sharepoint 2010必备组件及产品)
Reporting Service服务SharePoint集成模式安装配置 第三步和第四部 第三步 安装sharepoint 2010必备组件 1.安装SharePoint2010必备组件,执行Pre ...
- Win RT Webview获取cookie
方法1: HttpBaseProtocolFilter filter = new HttpBaseProtocolFilter(); var cookis = filter.CookieManager ...