题面描述
对于一个给定的正整数 n ,请你找出一共有多少种方式使 n 表示为若干个连续正整数的和,要求至少包括两个正整数。如 n=15 时,可以有 3 种方式:( 1+2+3+4+5 ),( 4+5+6 ),( 7+8 )。
输入数据
输入数据第一行为一个正整数 T ,表示测试数据的组数。 随后的 T 行中,每行包括一组测试数据,为一个整数 n(1≤T≤1000,n≤10^9)。
输出数据
对每一组输入数据,输出一行结果 ”Case #id: M” ,表示第 id 组数据的结果是 M , id 从 1 开始。
样例输入
2 |
样例输出
Case #1: 1 |
方法一:
思路:对于一个给定的数n,例如22,设置两个指针 i,j 则所求为 sum(i,j) = i+(i+1)+(i+2)+...j =n 其每一个数必定小于等于n/2 +1 ,若非如此,无论怎么加都会超过原来的数。 那么就可以循环判断 sum(i,j) 与 n 的关系:
当 sum(i,j) ==n 时,得到一个个结果。
当 sum (i, j) < n 时,说明还不够,j往后一位变成j+1,此时sum = sum + j +1。
当 sum(i,j) > n时,说明过了,此时i往后移一位,变成i+1。此时 sum = sum -i。
直到 i >= j 或者 j > n/2+1。
1 #include<iostream> 2 3 using namespace std; 4 5 6 int function(int n) 7 { 8 int mid = n / 2 + 1; 9 int i = 1; 10 int j = i + 1; 11 int sum = i + j; 12 int count = 0; 13 14 while (i < j && j <= mid) 15 { 16 if (sum == n) 17 { 18 count++; 19 sum -= i; 20 i++; 21 } 22 else if (sum < n) 23 { 24 j++; 25 sum += j; 26 } 27 else if (sum > n) 28 { 29 sum -= i; 30 i++; 31 } 32 } 33 return count; 34 } 35 36 int main() 37 { 38 int i, n; 39 cin >> n; 40 int *a = new int[n + 1]; 41 42 for (i = 1; i <= n; ++i) 43 { 44 cin >> a[i]; 45 } 46 47 for (i = 1; i <= n; ++i) 48 { 49 cout << "Case #" << i << ": " << function(a[i]) << endl; 50 } 51 52 return 0; 53 54 }
此程序效率极低。。。提交后超出时间限制。。。
方法二:
利用等差数列
an = a1+(n-1)d 这里公差d = 1
所以 an = a1+n-1
Sn = (a1+an)n/2 = (2a1+n-1)n/2
我们输入的数就是Sn ,我们要找的是以a1开始递增的数列使其和为Sn
这里我们可以用循环来判定,给定一个n,Sn已知,就可以求出a,如果a为正整数那么就可以找到等差数列的首项,加上n给定,d=1,那么就可以写出这个和式子。
注意,这里的n无须一直从2开始枚举下去,可以由Sn = (a1+an)n/2 = (2a1+n-1)n/2,所以a1=Sn/n-n/2+1/2,该式子为递减函数,n越大,a越小,而a最小为1,故另a=1时可确定n的最大范围。
令a=1,得二元一次方程(1/2+n/2)*n=Sn ,即n^2+n-2*Sn =0,可得方程两个根中取较大的根n=0.5*(-1+sqrt(1+8*Sn )),从而确定n的最大枚举范围。
#include<iostream> #include<math.h> using namespace std; int function(int n) { int count = 0; // count用来保存最后的结果 double a1; int N = (int)(0.5*(-1 + sqrt(1 + 8 * n))); for (int cnt = 2; cnt <= N; ++cnt) { 从2开始枚举 a1 = 1.0*n / cnt - cnt / 2.0 + 1 / 2.0; if ((int)a1 == a1 && a1 > 0) { // 如果得到的a1为整数且a1大于0 ,那么就符合预期 count++; // 每有一组符合,就加1 } } return count; } int main() { int i, n; cin >> n; int *a = new int[n + 1]; for (i = 1; i <= n; ++i) { cin >> a[i]; } for (i = 1; i <= n; ++i) { cout << "Case #" << i << ": " << function(a[i]) << endl; } system("pause"); return 0; }