Problem Description 题目链接
一个正整数一般可以分为几个互不相同的自然数的和,如3=1+2,4=1+3,5=1+4=2+3,6=1+5=2+4,…。现在你的任务是将指定的正整数n分解成若干个互不相同的自然数的和,且使这些自然数的乘积最大。
Input
只一个正整数n,(3<= n< 10000)。
Output
第一行是分解方案,相邻的数之间用一个空格分开,并且按由小到大的顺序。 第二行是最大的乘积。
Sample Input
10
Sample Output
2 3 5
30
#include <iostream> #include <math.h> #include <vector> #include <deque> #include <stdio.h> using namespace std; /* 思路:要使数尽可能多且每个数不能一样,用贪心思想从2-3-4-5开始分,直到不可分割之后,从最大的数往前累加到1,因为数字较大使用大数计算 比如:24 一开始为 2 3 4 5 6 余4 累加一轮为 2 4 5 6 7 */ //大数相乘 注意位置对齐即可 vector<int> BigDateMul(vector<int> nums, int n) { int index = 0; vector<int> temp; while (n) { int dig = 0, num = n % 10; for (int i = 0; i < nums.size(); i++) { //上一步的进位 dig += num * nums[i]; if (index + i >= temp.size()) { //说明超了 temp.push_back(dig % 10); } else { dig += temp[i + index]; temp[i + index] = dig % 10; } dig /= 10; }
if (dig) { temp.push_back(dig); } n /= 10; index++; } return temp; } int main() { int n; while (scanf("%d", &n) != EOF) { if (n == 3) { cout << "1 2 2 "; } else if (n == 4) { cout << "1 3 3 "; } else { //存放所有的数字 vector<int> allNum; int k = 2; while (k <= n) { allNum.push_back(k); n -= k; k++; } //从后向前累加 可能有几轮 while (n) { for (int i = allNum.size() - 1; i >= 0 && n; i--, n--) { allNum[i]++; } } //存放最后的大数 vector<int> nums; int num = allNum.back(); //转换成数组 while (num) { nums.push_back(num % 10); num /= 10; } //相乘 for (int i = allNum.size() - 2; i >= 0; i--) { nums = BigDateMul(nums, allNum[i]); } printf("%d", allNum.front()); for (int i = 1; i < allNum.size(); i++) { printf(" %d", allNum[i]); } printf(" "); for (int i = nums.size() - 1; i >= 0; i--) { printf("%d", nums[i]); } printf(" "); } } }