Luogu - P1018 乘积最大 - 题解
原文:https://www.luogu.org/problemnew/solution/P1018?page=7
题目:P1018【乘积最大】
前言:
- 这题的正解理论上说是DP,可是由于民间数据太水,用暴力过并不难
整体思路:
- 利用一个b数组标记每一位之间是否分割(1位分割,0为连接)。
- 利用STL里的 next_permutation 求出b的各种排列(即暴力枚举每种情况)。
- 由于本题数据规模大,所以要使用高精度计算每种分割的最后结果,并找出最大。
next_permutation函数:
- 即STL里的求全排列函数,所求的数组必须是升序,否则将无法求出全部的排列方式(这和它生成群排列的方式有关),next_permutation正常和sort一样,有2个参数,分别是数组的首地址和尾地址,并返回一个bool量,即能否求出下一个全排列,可以的话返回true,并将指定数组变为下一个排列方式,如1 2 3的下一个排列方式就是 1 3 2。
上代码:
#include<algorithm> //使用next_permutation需要调用的头文件
#include<cstdio> //c语言读入输出
#include<cstring> //处理高精度字符串时需要用到 using namespace std; struct BigN{ //高精度(即大整数)运算
int num[]={},len;
BigN(char s[]) //构造函数,用于给新定义的大整数赋值
{
len=strlen(s);
for(int i=len-;i>=;i--)
num[i]=s[len-i-]-'';
}
void clean() //用于清零
{
memset(num,,sizeof(num));
}
void f(int n) //将一个普通整数压到大整数的开头,这个在后面分割每一位时会用到
{
for(int i=len;i>;i--)
num[i]=num[i-];
len++;
num[]=n;
}
void cheng(BigN n)//高精度乘法,这里就不过多解释了,有疑问可以前往 P1303 了解更多
{
BigN c("");
int s=,g=;
for(int i=;i<=len;i++)
for(int j=;j<=n.len;j++)
{
int w=i+j;
s=num[i]*n.num[j];
c.num[w]+=s%;
c.num[w+]+=s/+c.num[w]/;
c.num[w]%=;
}
c.len=len+n.len;
while(c.num[c.len]==&&c.len>=)c.len--;
fz(c);
}
void fz(BigN n) //将一个大整数赋值给例外一个大整数,相当于'='
{
len=n.len;
for(int i=;i<=n.len;i++)
num[i]=n.num[i];
}
bool bj(BigN n) //判断两个大整数的大小,用于找出最大结果
{
if(len>n.len)
return ;
else if(len<n.len)
return ;
else
{
for(int i=len;i>=;i--)
if(num[i]<n.num[i])
return ;
else if(num[i]>n.num[i])
return ;
return -;
}
}
void out() //输出
{
for(int i=len;i>=;i--)
printf("%d",num[i]);
}
}; int n,k,sum[],b[],i,j; //常规定义,不多做解释
BigN mmax(""); int main()
{
char s[]; //s用于读入一个大整数
scanf("%d%d%s",&n,&k,&s);
for(i=;i<strlen(s);i++) //在sum中备份一份原数
sum[i]=s[i]-'';
for(i=n-;i>=(n-k)-;i--) //将b数组中的后k个数赋1,因为使用next_permutation需要让数组升序,否则可能无法找出所有排列方式
b[i]=;
do{
BigN temp(""),all("");//temp用于存放分割后的每一节,all用于计算每种排列方式的结果
i=;
while(i<n)//分割
{
if(i!=)
if(b[i-]==)//如果b[i-1]为1,那么就要在这一位加上一个乘号,即将原数分割
all.cheng(temp),temp.clean();//总数乘上分割后的每一位,并将temp清空,用于储存下一节.
temp.f(sum[i]),i++; //将原数的下一位压到temp的最前面
}
all.cheng(temp);//由于temp还没有乘all就退出循环,所以要再乘一次
if(mmax.bj(all)==)//如果这种排列顺序的结果大于之前最大的结果,刷新最大结果
mmax.fz(all);
}while(next_permutation(b,b+n-));//调用next_permutation
mmax.out();//输出
return ;
}
Luogu - P1018 乘积最大 - 题解的更多相关文章
- luogu P1018 乘积最大
题目描述 今年是国际数学联盟确定的"2000――世界数学年",又恰逢我国著名数学家华罗庚先生诞辰90周年.在华罗庚先生的家乡江苏金坛,组织了一场别开生面的数学智力竞赛的活动,你的一 ...
- P1018 乘积最大(高精度加/乘)
P1018 乘积最大 一道dp题目.比较好像的dp题目. 然而他需要高精度计算. 所以,他从我开始学oi,到现在.一直是60分的状态. 今天正打算复习模板.也就有借口解决了这道题目. #include ...
- 洛谷 P1018 乘积最大
P1018 乘积最大 题目描述 今年是国际数学联盟确定的“ 20002000 ――世界数学年”,又恰逢我国著名数学家华罗庚先生诞辰 9090 周年.在华罗庚先生的家乡江苏金坛,组织了一场别开生面的数学 ...
- P1018 乘积最大(DP)
题目 P1018 乘积最大 解析 区间DP 设\(f[i][j]\)表示选\(i\)个数,插入\(j\)个乘号时的最大值 设\(num[i][j]\)是\(s[i,j]\)里的数字 转移方程就是\(f ...
- 【luogu P5022 旅行】 题解
题目连接:https://www.luogu.org/problemnew/show/P5022 \(NOIP2018 DAY2T1\) 考场上只写了60分,很容易想到当 m = n - 1 时的树的 ...
- 洛谷P1018乘积最大——区间DP
题目:https://www.luogu.org/problemnew/show/P1018 区间DP+高精,注意初始化和转移的细节. 代码如下: #include<iostream> # ...
- 洛谷—— P1018 乘积最大
https://www.luogu.org/problem/show?pid=1018#sub 题目描述 今年是国际数学联盟确定的“2000――世界数学年”,又恰逢我国著名数学家华罗庚先生诞辰90周年 ...
- P1018 乘积最大
开始定义状态f[i][j][k]为[i,j)区间插入k个括号,使用记忆化搜索,但是成功爆栈,得到4个mle #include <bits/stdc++.h> using namespace ...
- 【luogu P2831 愤怒的小鸟】 题解
题目链接:https://www.luogu.org/problemnew/show/P2831 写点做题总结:dp,搜索,重在设计状态,状态设的好,转移起来也方便. 对于一条抛物线,三点确定.(0, ...
随机推荐
- Android Scrollview嵌套下listView动态加载数据,解决onScrollChanged执行多次数据重复问题
这一篇博客和上一篇讲的都是listView的动态加载,但有所不同的是,本篇的listView是嵌套在ScrollView下的,有时候在一个Activity中可能分为好几个模块,由于展示的需要(手机屏幕 ...
- Hyper-v,装XP的时候没有驱动上不了网,装这个集成服务(vmguest.iso )就可以了
Win10自带的Hyper-v,装XP的时候没有驱动上不了网,装这个集成服务(vmguest.iso )就可以了 安装后无法识别显卡及网卡设备,不能与虚拟网络通讯,设备管理器中显示三个未知设备. 在X ...
- 05. redis事务
目录 Redis 事务 事务 1. 命令有序 2. 始终原子 开启使用事务 Redis事务中出现错误 1. EXEC前的错误 2. EXEC后的错误 为什么出错了不支持roll backs? Redi ...
- centos6升级python版本至python3.5
一. 从Python官网到获取Python3的包, 切换到目录/usr/local/src wget https://www.python.org/ftp/python/3.5.1/Python-3. ...
- Linux的yum管理
前面介绍了软件的管理的方式rpm.但有个缺点,rpm不能解决依赖. 下面介绍的yum软件管理.可以完美的解决这个问题. 使用yum的方式管理rpm软件 优势:自动解决软件的依赖关系 ...
- SpringBoot 自定义线程池,多线程
原文:https://www.jianshu.com/p/832f2b162450 我们都知道spring只是为我们简单的处理线程池,每次用到线程总会new 一个新的线程,效率不高,所以我们需要自定义 ...
- docker更改默认存储路径
systemctl stop docker mkdir /data/docker cp -r /var/lib/docker/* /data/docker mv /var/lib/docker /va ...
- JMeter5.1开发SMTP协议接口脚本
jmeter可以测试发邮件和读取邮件. 发送邮件 上图部分解释: Server:邮件发送服务 Port:发邮件端口,不加密25,加密465,如果是465端口,Security settings 需要选 ...
- springboot2.0整合logback日志(详细)
<div class="post"> <h1 class="postTitle"> springboot2.0整合logback日志(详 ...
- three.js 添加环境光
var ambient = new THREE.AmbientLight(0xffffff);//环境光 scene.add(ambient);