编程之美:第二章 数字之魅 2.21只考加法的面试题

/*
只考加法的面试题:
1+2=3
4+5=9
2+3+4=9
写一个程序,对于64位正整数,输出它所有可能的连续自然数(两个以上)之和的形式

思路:
可以采用暴力求解。以上限n/2+1 + n/2作为起始点,如果满足,输出;不满足时肯定是偏大,那么使n/2向左移动,再次计算,如果仍然偏大,就向左移动n/2+1;
偏小向左移动左边指针,偏大向左移动右边指针,如果当左边移动到1处循环截止

输入:
9
21
10
输出:
4+5=9
2+3+4=9

10+11=21
6+7+8=21
1+2+3+4+5+6=21

1+2+3+4=10
*/
#include <stdio.h>

void successiveNumAdd(long long n)
{
	static int iCnt = 0;
	if(n <= 0)
	{
		return;
	}
	long long iLow = n/2;
	long long iHigh = n/2+1;
	if((n & 1) == 0)//如果是偶数,就将iLow与iHigh都减1
	{
		iLow--;
		iHigh--;
	}
	int iSum;
	while(iLow > 0)//iLow == 0时,表示循环结束
	{
		iSum = ((iLow + iHigh)*(iHigh - iLow + 1))/2;//;利用等差公式求和
		if(iSum > n )//如果所求的和大于原来的数,让iHigh--(这样会减小)
		{
			iHigh--;
		}
		else if(iSum < n)//如果所求的和小于原来的数,让iLow--(这样会增大)
		{
			iLow--;
		}
		else//我们知道一旦等于以后,后面应该让iHigh--,使其减小,然后后面通过iLow--再使数增大起来
		{
			for(int i = iLow ; i <= iHigh ; i++)
			{
				if(i != iLow)
				{
					printf("+%d",i);
				}
				else
				{
					printf("%d",i);
				}
			}
			printf("=%d\n",n);
			iHigh--;
			iCnt++;
		}
	}
	printf("%d\n",iCnt);
}

void process()
{
	long long n;
	while(EOF != scanf("%lld",&n))
	{
		successiveNumAdd(n);
	}
}

int main(int argc,char* argv[])
{
	process();
	getchar();
	return 0;
}

你可能感兴趣的:(编程之美)