原文:https://www.luogu.org/problemnew/solution/P1018?page=7

题目:P1018【乘积最大】


前言:

  • 这题的正解理论上说是DP,可是由于民间数据太水,用暴力过并不难

整体思路:

  1. 利用一个b数组标记每一位之间是否分割(1位分割,0为连接)。
  2. 利用STL里的 next_permutation 求出b的各种排列(即暴力枚举每种情况)。
  3. 由于本题数据规模大,所以要使用高精度计算每种分割的最后结果,并找出最大。

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 乘积最大 - 题解的更多相关文章

  1. luogu P1018 乘积最大

    题目描述 今年是国际数学联盟确定的"2000――世界数学年",又恰逢我国著名数学家华罗庚先生诞辰90周年.在华罗庚先生的家乡江苏金坛,组织了一场别开生面的数学智力竞赛的活动,你的一 ...

  2. P1018 乘积最大(高精度加/乘)

    P1018 乘积最大 一道dp题目.比较好像的dp题目. 然而他需要高精度计算. 所以,他从我开始学oi,到现在.一直是60分的状态. 今天正打算复习模板.也就有借口解决了这道题目. #include ...

  3. 洛谷 P1018 乘积最大

    P1018 乘积最大 题目描述 今年是国际数学联盟确定的“ 20002000 ――世界数学年”,又恰逢我国著名数学家华罗庚先生诞辰 9090 周年.在华罗庚先生的家乡江苏金坛,组织了一场别开生面的数学 ...

  4. P1018 乘积最大(DP)

    题目 P1018 乘积最大 解析 区间DP 设\(f[i][j]\)表示选\(i\)个数,插入\(j\)个乘号时的最大值 设\(num[i][j]\)是\(s[i,j]\)里的数字 转移方程就是\(f ...

  5. 【luogu P5022 旅行】 题解

    题目连接:https://www.luogu.org/problemnew/show/P5022 \(NOIP2018 DAY2T1\) 考场上只写了60分,很容易想到当 m = n - 1 时的树的 ...

  6. 洛谷P1018乘积最大——区间DP

    题目:https://www.luogu.org/problemnew/show/P1018 区间DP+高精,注意初始化和转移的细节. 代码如下: #include<iostream> # ...

  7. 洛谷—— P1018 乘积最大

    https://www.luogu.org/problem/show?pid=1018#sub 题目描述 今年是国际数学联盟确定的“2000――世界数学年”,又恰逢我国著名数学家华罗庚先生诞辰90周年 ...

  8. P1018 乘积最大

    开始定义状态f[i][j][k]为[i,j)区间插入k个括号,使用记忆化搜索,但是成功爆栈,得到4个mle #include <bits/stdc++.h> using namespace ...

  9. 【luogu P2831 愤怒的小鸟】 题解

    题目链接:https://www.luogu.org/problemnew/show/P2831 写点做题总结:dp,搜索,重在设计状态,状态设的好,转移起来也方便. 对于一条抛物线,三点确定.(0, ...

随机推荐

  1. tcp / udp 协议及其实现的socket

    一.tcp协议 1.1 基本知识 特点: 可靠,慢,全双工通信 建立连接时:三次握手 断开连接时:四次挥手 在建立起连接之后 发送的每一条信息都有回执 为了保证数据的完整性,还有重传机制 长连接:会一 ...

  2. 使用gacutil把COM组件注册到全局缓存GAC中

    我们在编写软件的时候,有时候需要调用COM组件,那就需要注册了,注册有两种,一种是使用regasm 在程序运行的时候注册,参考“pb调用C#编写的DLL类库“,不过受路径的限制.还有一种注册方式,使用 ...

  3. 海思3519 qt ffmpeg 软解码播放avi

    在海思3519上基于qt采用ffmpeg对avi进行解码显示,其中ffmpeg的配置,qt的配置在前文中已经说明,在此不再赘述. 解码 解码在单独的线程中进行,具体的代码如下: void VideoP ...

  4. sql字符转义

    /** * sql 内的like % 代表一个或多个字符, _代表一个字符,这都是需要转义的 * @param s * @return */ public static String escapeSp ...

  5. 【Tomcat】使用Eclipse绑定Tomcat 发布应用&&常见错误

     创建时间:6.14 一.Eclipse绑定Tomcat 步骤1:获得服务器运行环境配置,Window/Preferences/Server/Runtime Environment 步骤2:添加服务器 ...

  6. django的form 登录组件

    1. 了解form 组件的原理 1.建立好form组件 class>>> 2.了解需要先is_valid() 判断是否可以取值 成功 form.cleaned_data 查看成功认证 ...

  7. Kubernetes部署Spring Boot应用

    SpringBoot项目 新建springboot项目 @RestController public class HelloWorldController { @RequestMapping(&quo ...

  8. el获取对象属性大小写问题

    当el表达式首字母大写,会判断第二个字母大小写,小写,抛异常:大写,则直接拼接get,形成getter方法 当el表达式首字母小写,会判断第二个字母大小写,大写,必须第一个字母大写:否则无法获取值 e ...

  9. Python面向对象 | isinstance和issubclass

    isinstance(a,b):判断a是否是b类(或者b类的基类)实例化的对象 class A: pass class B(A): pass obj = B() print(isinstance(ob ...

  10. qt ubutun各个版本下载地址

    http://download.qt.io/archive/qt/ http://mirrors.melbourne.co.uk/ubuntu-releases/