P1018 [NOIP2000 提高组] 乘积最大

题目

题目链接

题解

状态定义:dp[i][j]表示前i个数分成j段(即需要j+1*)的最大乘积。
状态转移:dp[i][j] = max(dp[k-1][j-1]*a[k][i], dp[i][j]),表示在第k-1和第k个数之间加上一个*得到的最大值,其中前k-1个数采用的是dp值,即被分为了j-1段后的最大值,而后面的数(从ki)则视为一整个数,与前面的dp值相乘。遍历全部k的位置,取最大值即为 dp[i][j]


首先,暴力是不行的,暴力枚举每个*的位置,O(40^6)=O((2^12)*(10^6))=O(4,096,000,000)

再者,由于要保存和计算的数最高为40位,因此还需要采用高精度。(之前写的高精度,有空重新整理一下)


整体思路就是先写出来非高精度版,再将其中的乘、加改成高精度的函数即可。

高精度函数:

  1. max:计算两个高精度数的较大者,并返回string类型的高精度数。
  2. add:计算一个高精度数和一个int类型数的和,因为用不到两个高精度数相加,所以就写了这一个。
  3. mul:重载成两个函数了。第一个是计算两个高精度数的乘积,返回string类型的高精度数;第二个是计算一个高精度数与10相乘的结果。

另:如果在提交时出现RE,或者测试我给出的样例时出现了奇怪错误,要么是数组开小了,要么是高精度乘法最后位数的处理出现了问题。(经验之谈)

代码

未采用高精度的60%代码:

#include<bits/stdc++.h>
using namespace std;
const int N = 50, M = 10;

int n, m;
long long dp[N][M], a[N][N];
string num;

int main()
{
	cin >> n >> m;
	cin >> num;
	
	for (int i = 0;i < n;i ++)
		for (int j = i;j < n;j ++)
			a[i+1][j+1] = a[i+1][j]*10 + (num[j]-'0');
	
	for (int i = 1;i <= n;i ++)
		dp[i][1] = a[1][i];
	
	for (int i = 1;i <= n;i ++)
		for (int j = 2;j <= m+1 && j <= i;j ++) 
			for (int k = j;k <= i;k ++)
				dp[i][j] = max(dp[k-1][j-1] * a[k][i], dp[i][j]);
		
	cout << dp[n][m+1] << endl;
	
	return 0;
}

采用高精度的AC代码:

#include<bits/stdc++.h>
using namespace std;
const int N = 100, M = 20;

int n, m;
long long dp[N][M], a[N][N];
string num, f[N][M], b[N][N];

string max (string s1, string s2) {
	// 先比较长度,长者大;否则比较高位数,大者大;否则一样。
	int n1 = s1.size(), n2 = s2.size();
	if (n1 != n2) return n1>n2?s1:s2;
	for (int i = 0;i < n1;i ++)
		if (s1[i] != s2[i]) 
			return s1[i] > s2[i] ? s1 : s2;
	return s1; 
}

string add (string s, int c) {
	int nn = s.size();
	for (int i = nn-1;i >= 0;i --) {
		int t = s[i] - '0';
		s[i] = (t + c) % 10 + '0';
		c = (t + c) / 10;
	}
	return (c == 0 ? s : "1" + s);
}

string mul (string s1, string s2) {
	// 比较标准的高精度加法模板
	int len1 = s1.size(), len2 = s2.size();
	int c[len1+len2+1], a[len1+1], b[len2+1];
	s1 = '.'+s1; s2 = '.'+s2;
	memset(c, 0, sizeof c);
	
	for(int i=1;i<=len1;i++)
        a[len1-i+1]=s1[i]-'0';
    for(int i=1;i<=len2;i++)
        b[len2-i+1]=s2[i]-'0';
 
    for(int i=1;i<=len1;i++)   
        for(int j=1;j<=len2;j++){
            c[i+j-1]+=a[i]*b[j];
            c[i+j]+=c[i+j-1]/10;
            c[i+j-1]%=10;
        }
 
    int len=len1+len2-1;
    if(c[len+1]!=0) len++;
 
 	string res = "";
    for(int i=len;i>=1;i--) {
    	res += to_string(c[i]); // to_string将int转为string,string只能拼接string,不能拼接char
	}
    return res;

	/*
	// 这种方式也可以,不标准
	reverse(s1.begin(), s1.end());
	reverse(s2.begin(), s2.end());
	int n1 = s1.size(), n2 = s2.size(), c = 0;
	int res[n1+n2+1], len = n1 + n2 - 1;
	memset(res, 0, sizeof res);
	
	for (int i = 0;i < n1;i ++)
		for (int j = 0;j < n2;j ++) {
			int t = (s1[i]-'0') * (s2[j]-'0');
			res[i+j] += t;
			res[i+j+1] += res[i+j]/10;
			res[i+j] %= 10;
		}
	
	while(res[len]) len ++; // 将len初始化为n1+n2-1是因为两个数乘积最少位数为两个数位数之和-1
	string ans(len, '0'); // 必须这样初始化,否则直接将字符赋值到string上进行输出是空的
	for (int i = 0;i < len;i ++) 
		ans[len-i-1] = res[i] + '0';
	return ans;
	*/ 
}

string mul (string s) {
	// 默认乘以10
	// 乘以10,直接加个0就行 
	return s + "0" ;
}

// main部分的代码几乎没变,就是加了几个函数、改了几个变量名
int main()
{
	cin >> n >> m;
	cin >> num;
	
	for (int i = 0;i < n;i ++)
		for (int j = i;j < n;j ++)
			b[i+1][j+1] = add(mul(b[i+1][j]), num[j]-'0');
			
	for (int i = 1;i <= n;i ++)
		f[i][1] = b[1][i];
	
	for (int i = 1;i <= n;i ++)
		for (int j = 2;j <= m+1 && j <= i;j ++) 
			for (int k = j;k <= i;k ++) {
				f[i][j] = max(mul(f[k-1][j-1], b[k][i]), f[i][j]);		
			}

		
	cout << f[n][m+1] << endl;
	
	return 0;
}

/*
几组样例
 
15 5
234567891987127
93730005427968

40 5
2323456789194567892345678919871271987127
1799210433424844361866616035388793782468

40 5
2323456789194567892345678919871271900000
1658633212909120759464865210877632800000
*/
  • 5
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 1
    评论
题目描述 给出一单词,按照以下规则进行单词接龙: 1.首先任意选定一个单词作为开头; 2.接下来每次接的单词必须满足:它的首字母与上一个单词的尾字母相同; 3.同一个单词不能重复使用; 4.当无法继续接下去时,游戏结束。 你的任务是编写一个程序,计算出在给定单词集合中,最长的单词接龙长度。 输入格式 输入文件的第一行为一个整数n,表示单词的个数。 接下来n行,每行一个单词,由小写字母成,长度不超过20个字符。 输出格式 输出文件仅一行,包含一个整数,表示最长的单词接龙长度。 数据范围 1≤n≤500 输入样例 5 ab bc cd de ef 输出样例 5 提示 输入样例2: 6 a b c d e f 输出样例2: 1 输入样例3: 4 aa bb cc dd 输出样例3: 2 输入样例4: 4 a b ba ab 输出样例4: 2 输入样例5: 3 a aa aaa 输出样例5: 1 输入样例6: 3 a aa ab 输出样例6: 2 输入样例7: 4 a aa aaa aaaa 输出样例7: 1 输入样例8: 4 a aa aaa baaa 输出样例8: 2 输入样例9: 4 a aa baa ab 输出样例9: 2 输入样例10: 4 a aa baa ba 输出样例10: 2 输入样例11: 4 a aa baa bb 输出样例11: 2 输入样例12: 4 a aa baa bc 输出样例12: 2 输入样例13: 4 a aa baa bca 输出样例13: 2 输入样例14: 4 a aa baa bcb 输出样例14: 2 输入样例15: 4 a aa baa bcc 输出样例15: 2 输入样例16: 4 a aa baa bcdc 输出样例16: 2 输入样例17: 4 a aa baa bcde 输出样例17: 2 输入样例18: 4 a aa baa bcdef 输出样例18: 2 输入样例19: 4 a aa baa bcdefg 输出样例19: 2 输入样例20: 4 a aa baa bcdefgh 输出样例20: 2 输入样例21: 4 a aa baa bcdefghi 输出样例21: 2 输入样例22: 4 a aa baa bcdefghij 输出样例22: 2 输入样例23: 4 a aa baa bcdefghijk 输出样例23: 2 输入样例24: 4 a aa baa bcdefghijkl 输出样例24: 2 输入样例25: 4 a aa baa bcdefghijklm 输出样例25: 2 输入样例26: 4 a aa baa bcdefghijklmn 输出样例26: 2 输入样例27: 4 a aa baa bcdefghijklmno 输出样例27: 2 输入样例28: 4 a aa baa bcdefghijklmnop 输出样例28: 2 输入样例29: 4 a aa baa bcdefghijklmnopq 输出样例29: 2 输入样例30: 4 a aa baa bcdefghijklmnopqr 输出样例30: 2 输入样例31: 4 a aa baa bcdefghijklmnopqrs 输出样例31: 2 输入样例32: 4 a aa baa bcdefghijklmnopqrst 输出样例32: 2 输入样例33: 4 a aa baa bcdefghijklmnopqrstu 输出样例33: 2 输入样例34: 4 a aa baa bcdefghijklmnopqrstuv 输出样例34: 2 输入样例35: 4 a aa baa bcdefghijklmnopqrstuvw 输出样例35: 2 输入样例36: 4 a aa baa bcdefghijklmnopqrstuvwx 输出样例36: 2 输入样例37: 4 a aa baa bcdefghijklmnopqrstuvwxy 输出样例37: 2 输入样例38: 4 a aa baa bcdefghijklmnopqrstuvwxyz 输出样例38: 2 输入样例39: 4 a aa baa bcdefghijklmnopqrstuvwxyzA 输出样例39: 2 输入样例40: 4 a aa baa bcdefghijklmnopqrstuvwxyzAB 输出样例40: 2 输入样例41: 4 a aa baa bcdefghijklmnopqrstuvwxyzABC 输出样例41: 2 输入样例42: 4 a aa baa bcdefghijklmnopqrstuvwxyzABCD 输出样例42: 2 输入样例43: 4 a aa baa bcdefghijklmnopqrstuvwxyzABCDE 输出样例43: 2 输入样例44: 4 a aa baa bcdefghijklmnopqrstuvwxyzABCDEF 输出样例44: 2 输入样例45: 4 a aa baa bcdefghijklmnopqrstuvwxyzABCDEFG 输出样例45: 2 输入样例46: 4 a aa baa bcdefghijklmnopqrstuvwxyzABCDEFGH 输出样例46: 2 输入样例47: 4 a aa baa bcdefghijklmnopqrstuvwxyzABCDEFGHI 输出样例47: 2 输入样例48: 4 a aa baa bcdefghijklmnopqrstuvwxyzABCDEFGHIJ 输出样例48: 2 输入样例49: 4 a aa baa bcdefghijklmnopqrstuvwxyzABCDEFGHIJK 输出样例49: 2 输入样例50: 4 a aa baa bcdefghijklmnopqrstuvwxyzABCDEFGHIJKL 输出样例50: 2 输入样例51: 4 a aa baa bcdefghijklmnopqrstuvwxyzABCDEFGHIJKLM 输出样例51: 2 输入样例52: 4 a aa baa bcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMN 输出样例52: 2 输入样例53: 4 a aa baa bcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNO 输出样例53: 2 输入样例54: 4 a aa baa bcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOP 输出样例54: 2 输入样例55: 4 a aa baa bcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQ 输出样例55: 2 输入样例56: 4 a aa baa bcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQR 输出样例56: 2 输入样例57: 4 a aa baa bcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRS 输出样例57: 2 输入样例58: 4 a aa baa bcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRST 输出样例58: 2 输入样例59: 4 a aa baa bcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTU 输出样例59: 2 输入样例60: 4 a aa baa bcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUV 输出样例60: 2 输入样例61: 4 a aa baa bcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVW 输出样例61: 2 输入样例62: 4 a aa baa bcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWX 输出样例62: 2 输入样例63: 4 a aa baa bcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXY 输出样例63: 2 输入样例64: 4 a aa baa bcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ 输出样例64: 2 输入样例65: 4 a aa baa bcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZA 输出样例65: 2 输入样例66: 4 a aa baa bcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZAB 输出样例66: 2 输入样例67: 4 a aa baa bcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZABC 输出样例67: 2 输入样例68: 4 a aa baa bcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZABCD 输出样例68: 2 输入样例69: 4 a aa baa bcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZABCDE 输出样例69: 2 输入样例70: 4 a aa baa bcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZABCDEF 输出样例70: 2 输入样例71: 4 a aa baa bcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZABCDEFG 输出样例71: 2 输入样例72: 4 a aa baa bcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZABCDEFGH 输出样例72: 2 输入样例73: 4 a aa baa bcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZABCDEFGHI 输出样例73: 2 输入样例74: 4 a aa baa bcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZABCDEFGHIJ 输出样例74: 2 输入样例75: 4 a aa baa bcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZABCDEFGHIJK 输出样例75: 2 输入样例76: 4 a aa baa bcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZABCDEFGHIJKL 输出样例76: 2 输入样例77: 4 a aa baa bcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZABCDEFGHIJKLM 输出样例77: 2 输入样例78: 4 a aa baa bcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZABCDEFGHIJKLMN 输出样例78: 2 输入样例79: 4 a aa baa bcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZABCDEFGHIJKLMNO 输出样例79: 2 输入样例80: 4 a aa baa bcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZABCDEFGHIJKLMNOP 输出样例80: 2 输入样例81: 4 a aa baa bcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZABCDEFGHIJKLMNOPQ 输出样例81: 2 输入样例82: 4 a aa baa bcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZABCDEFGHIJKLMNOPQR 输出样例82: 2 输入样例83: 4 a aa baa bcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZABCDEFGHIJKLMNOPQRS 输出样例83: 2 输入样例84: 4 a aa baa bcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZABCDEFGHIJKLMNOPQRST 输出样例84: 2 输入样例85: 4 a aa baa bcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZABCDEFGHIJKLMNOPQRSTU 输出样例85: 2 输入样例86: 4 a aa baa bcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZABCDEFGHIJKLMNOPQRSTUV 输出样例86: 2 输入样例87: 4 a aa baa bcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZABCDEFGHIJKLMNOPQRSTUVW 输出样例87: 2 输入样例88: 4 a aa baa bcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZABCDEFGHIJKLMNOPQRSTUVWX 输出样例88: 2 输入样例89: 4 a aa baa bcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZABCDEFGHIJKLMNOPQRSTUVWXY 输出样例89: 2 输入样例90: 4

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

不牌不改

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值