原文: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. 你家的APS系统有这些功能吗?排程系统功能盘点

    随着企业规模不断扩大,在经营管理方面会面临各种各样的问题,为了帮助解决此类问题,很多公司都会引入APS高级排程系统帮助进行生产管理的优化. APS系统针对的管理目标是 改善库存控制,大幅降低原料与中间 ...

  2. elasticsearch查询篇索引映射文档数据准备

    elasticsearch查询篇索引映射文档数据准备 我们后面要讲elasticsearch查询,先来准备下索引,映射以及文档: 我们先用Head插件建立索引film,然后建立映射 POST http ...

  3. Linux Tomcat安装及端口配置

    1.  JDK安装配置 待写 2. Tomcat安装配置 1,下载Tomcat链接,到启动测试. 将文件apache-tomcat-8.5.50.tar.gz移动到/usr/tomcat/下,并解压 ...

  4. HBaseAPI

    环境准备 新建项目后在pom.xml中添加依赖: <dependency> <groupId>org.apache.hbase</groupId> <arti ...

  5. FileZilla 连接站点 编辑配置文件

  6. MacOS下IDEA设置智能提示不区分大小写

    本文只针对,IDEA-2019.2.3版本 目录地址: Edit -> General -> Code Completion -> Match case -> 勾选去掉 截图如 ...

  7. classmethode,staticmethode、反射

    目录 classmethod: staticmethod: classmethod与staticmethod都是python解释器内置的装饰器 类中定义的函数分为两大类:绑定方法和非绑定方法 在类中正 ...

  8. 论OIer的同构

    定义一个输入集,指这样一个集合:由任意存在于物理世界的OI题目以及做这个题的时间与身体状态等各种元素组成的n元组组成的集合 OIer被定义为一个二元组:<"结果集",&quo ...

  9. 【CSP-S膜你考】 A

    A 题面 对于给定的一个正整数n, 判断n是否能分成若干个正整数之和 (可以重复) , 其中每个正整数都能表示成两个质数乘积. 输入格式 第一行一个正整数 q,表示询问组数. 接下来 q 行,每行一个 ...

  10. shell equal

    #!/bin/shzero=0 status=1 let status=0 if [[ $status -eq $zero ]];then echo "bu >= 3.6"e ...