PAT1103
1103. Integer Factorization (30)
The K-P factorization of a positive integer N is to write N as the sum of the P-th power of K positive integers. You are supposed to write a program to find the K-P factorization of N for any positive integers N, K and P.
Input Specification:
Each input file contains one test case which gives in a line the three positive integers N (<=400), K (<=N) and P (1<P<=7). The numbers in a line are separated by a space.
Output Specification:
For each case, if the solution exists, output in the format:
N = n1^P + ... nK^P
where ni (i=1, ... K) is the i-th factor. All the factors must be printed in non-increasing order.
Note: the solution may not be unique. For example, the 5-2 factorization of 169 has 9 solutions, such as 122 + 42 + 22 + 22 + 12, or 112 + 62+ 22 + 22 + 22, or more. You must output the one with the maximum sum of the factors. If there is a tie, the largest factor sequence must be chosen -- sequence { a1, a2, ... aK } is said to be larger than { b1, b2, ... bK } if there exists 1<=L<=K such that ai=bi for i<L and aL>bL
If there is no solution, simple output "Impossible".
Sample Input 1:
169 5 2
Sample Output 1:
169 = 6^2 + 6^2 + 6^2 + 6^2 + 5^2
Sample Input 2:
169 167 3
Sample Output 2:
Impossible
这是一道很考验dfs能力的题目。思路很简单,找到所有的数i(i^p<n) 然后对这个数集进行dfs。
我们来先看第一种写法
void dfs(int index,int sum,int num,int factsum)
{ if(sum==n&&num==k)
{ if(factsum>maxsum)
{
maxsum=factsum;
ans=temp;
}
else if(factsum==maxsum)
{ for(int i=;i<temp.size();i++)
{
if(temp[i]>ans[i])
{
ans=temp;
break;
}
else if(temp[i]<ans[i])
break;
}
}
return;
}
if(num>k||sum>n)
return; for(int i=index;i<=t;i++)
{
temp.push_back(v[i]); dfs(index+,sum+w[i],num+,factsum+v[i]); //每次允许重复的数字一定比第一个数字大,并且可以重复的次数与重复的数字有关。
temp.pop_back(); }*/
return;
}
过了一部分样例,但是仔细分析发现这种方式重复的数字是有限制的,第一个数只能重复两次,后面的数如果想要重复必须比第一个数字大,并且重复的次数也有要求。
例如对样例18 4 2就无法得到1 2 2 3因为比1大的数2最多只能重复(2-index)次,这里index为2
再看下一种写法:
void dfs(int sum,int num,int factsum)
{
if(sum==n&&num==k)
{
if(factsum>maxsum)
{
maxsum=factsum;
ans=temp;
}
else if(factsum==maxsum)
{ for(int i=;i<temp.size();i++)
{
if(temp[i]>ans[i])
{
ans=temp;
break;
}
else if(temp[i]<ans[i])
break;
}
}
return;
}
if(num>k||sum>n)
return;
for(int i=;i<=t;i++)
{
temp.push_back(v[i]); dfs(sum+w[i],num+,factsum+v[i]); //每次允许重复的数字一定比第一个数字大,并且可以重复的次数与重复的数字有关。
temp.pop_back(); }
return;
}
这种方法固然可行,但是肯定会超时。每次都重1开始计算,会出现大量重复的情况
void dfs(int index,int sum,int num,int factsum)
{
if(sum==n&&num==k)
{
if(factsum>maxsum)
{
maxsum=factsum;
ans=temp;
}
else if(factsum==maxsum)
{ for(int i=;i<temp.size();i++)
{
if(temp[i]>ans[i])
{
ans=temp;
break;
}
else if(temp[i]<ans[i])
break;
}
}
return;
}
if(num>k||sum>n)
return;
if(index>=)
{
temp.push_back(index);
dfs(index,sum+w[index],num+,factsum+v[index]);
temp.pop_back();
dfs(index-,sum,num,factsum);
}
/*for(int i=index;i>=1;i--)
{
temp.push_back(v[i]); dfs(index-1,sum+w[i],num+1,factsum+v[i]); //每次允许重复的数字一定比第一个数字大,并且可以重复的次数与重复的数字有关。
temp.pop_back(); }*/
return;
}
正解如上。从后往前递推,每个数字都可以重复多次,如果不符合条件就减1。同时这样相等时,第一个序列即为题目要求的输出。从前往后仍然可能会超时,因为初始的数字过于小了。
网上题解:
为了顺应题目找到最大系数和或者最大系数列,我们从小到大进行枚举,这样即使碰到了和相等的情况,由于是递增着枚举的,因此直接覆盖原来的系数列,得到的就是最终满足条件的系数列。
我们利用DFS来从小到大的枚举,DFS函数的参数如下:
dfs(long long N, int cur, vector<int>& factors);
①其中N是当前值,从最初的输入开始,逐步减去每个系数的运算结果;cur是枚举到的位置,从0开始,依次填入factors容器中,当cur==K时,枚举已经结束,我们判断N是否为0,为0则找到了一个满足条件的系数列,并且这个系数列按照升序存储在factors中。注意到cur==K但N≠0是我们的第一个剪枝条件。
②为了保证枚举从小到大开始,在每次枚举开始前计算lower和upper两个值,其中lower由factors中刚刚枚举完的上一个值确定,如果当前是枚举的起始点,则从1开始;upper为根号下N,因为系数的次方P>1,因此最大的系数不可能超过根号N。
③对lower到upper内的每一个值,计算系数的P次方,用res表示。如果N≥res,则说明合法,将其填入factors中并且向后枚举,即
- factors[cur] = i;
- dfs(N-res,cur+1,factors);
如果N<res,说明系数偏大,又因为系数是递增枚举的,所以此后都不满足,直接返回,这是我们的第二个剪枝条件。
综合上面两个剪枝条件,即可写出高效的dfs算法来枚举结果,为了能得到满足题目要求的系数列,我们设置全局变量nowSum和finalFactor,每次找到一个系数列,就求其系数和sum,如果sum≥nowSum,则更新nowSum与finalFactor,等号涵盖了题目中的第二个条件,因为后面出现的系数列一定大于前面出现的系数列(递增枚举)。
最后如果finalFactor规模为K,则说明找到,先反转,后输出,注意格式;否则输出Impossible。
这个题解是典型的剪枝方法
#include <iostream>
#include <cstdio>
#include <cmath>
#include <vector>
#include <algorithm> using namespace std; typedef long long lint; lint N,K;
int P; lint lpower(lint n, lint p){
if(n == ) return ;
int factor = n;
for(int i = ; i < p; i++) n *= factor;
return n;
} vector<int> finalFactor;
int nowSum = ; bool dfs(lint N, int cur, vector<int>& factors){
if(cur == K){
if(N == ){
int sum = ;
for(int i = ; i < factors.size(); i++){
sum += factors[i];
}
if(sum >= nowSum){
finalFactor = factors;
nowSum = sum;
}
return true;
}else return false;
}
lint upper = sqrt((double)N);
lint lower = cur > ? factors[cur - ] : ;
for(lint i = lower; i <= upper; i++){
lint res = lpower(i,P);
if(N >= res){
factors[cur] = i;
dfs(N-res,cur+,factors);
}else{
return false;
}
}
return true;
} int main()
{
cin >> N >> K >> P;
vector<int> factors(K);
dfs(N,,factors);
reverse(finalFactor.begin(),finalFactor.end());
if(finalFactor.size() == K){
printf("%d = ",N);
printf("%d^%d",finalFactor[],P);
for(int i = ; i < finalFactor.size(); i++){
printf(" + %d^%d",finalFactor[i],P);
}
}else{
cout << "Impossible";
}
cout << endl; return ;
}
PAT1103的更多相关文章
随机推荐
- jsp取addFlashAttribute值深入理解即springMVC发redirect传隐藏参数
结论:两种方式 a.如果没有进行action转发,在页面中el需要${sessionScope['org.springframework.web.servlet.support.SessionFlas ...
- leetcode212
class Solution { public List<String> findWords(char[][] board, String[] words) { List<Strin ...
- leetcode997
class Solution: def findJudge(self, N: int, trust: 'List[List[int]]') -> int: if N==1 and len(tru ...
- 利用STM32CubeMX来生成USB_HID_host工程
修改时钟(备注这边使用25mhz的主晶振) 选择debug_level等级为3
- nginx配置资源缓存
缓存nginx服务器的静态文件.如css,js,htm,html,jpg,gif,png,flv,swf,这些文件都不是经常更新.便于缓存以减轻服务器的压力. 打开配置文件/usr/local/ngi ...
- requirements.txt 的使用与创建
1. requirements.txt 主要是记录你的python 解释器安装了那些第三方模块,这样好方便项目迁移,自动解决掉项目的依赖关系 2. 网上找的那些关于 requirements 的文档 ...
- Java 基础 - 对象池
对象池 优点: 防止过多的创建对象合理利用对象, 缺点: 会有线程阻塞 Demo 测试代码 package com.cjcx.pay.obj; import java.util.Enumerati ...
- HBuilder开发APP自动登录时跳过"登录页面"
刚接触开发公司APP项目,用HBuilder开发工具. manifest.json中的入口页面就是"登录页面",现在获取到自动登录状态是true,但是真机联调时"登录页面 ...
- 【Noip模拟 20161004】局域网
问题描述 所有SZSZ 学生翘首以盼的新教学楼总算快要竣工了,接下来到了网络布线的时候.网络系统的总布局是由nn台计算机组成的有线局域网,每根网线长度为dd,正常情况下,网线是可以缠绕使其变短但是不能 ...
- C语言复习:指针知识
指针知识体系搭建 指针强化 指针是一种数据类型 指针也是一种变量,占有内存空间,用来保存内存地址 测试指针变量占有内存空间大小:sizeof(指针名); 2)*p操作内存 在指针声明时,*号表示所声明 ...