第2章 循环结构程序设计

2.1 for循环

例题2.1 aabb(输出所有的形如aabb的4位完全平方数)
算法一:对1000到9999分别进行aabb以及完全平方的判断,但很明显效率较低。
#include <stdio.h>
#include <math.h>
int main()
{
	double square;
	for(int i = 1000; i <= 9999; i++){ //尽量缩短变量的定义范围,在for循环的初始化部分定义循环变量
		square = floor(sqrt(double(i)) + 0.5);
		if(i%10 == i%100/10 && i/100%10 == i/1000 && square*square == i)
			printf("%d ",i);
	}
	printf("\n");
	return 0;
}
算法二:进行双层循环,分别对a进行1到9和对b进行0到9的遍历,再对得到的形如aabb的四位数做完全平方的判断。注意四舍五入floor(sqrt(double(n)) + 0.5),如floor(x + 0.5)等于1的区间为[0.5, 1.5)。
#include <stdio.h>
#include <math.h>
int main()
{
	for(int a = 1; a < 10; a++){  //a从1开始,因为是四位数
		for(int b = 0; b < 10; b++){
			int n = a * 1100 + b *11;
			int m = floor(sqrt(double(n)) + 0.5);
			if(m*m == n) printf("%d ", n);
		}
	}
	printf("\n");
	return 0;
}
算法三:枚举平方根,从而避免开平方操作,square从1开始,当其平方数小于1000时开始下层循环,大于9999时跳出循环,循环体内进行aabb的判断。
#include <stdio.h>
#include <math.h>
int main()
{
	for(int square = 1; ; square++){
		int n = square*square;
		if(n < 1000) continue;
		if(n > 9999) break;
		if(n%10 == n%100/10 && n/100%10 == n/1000)
			printf("%d ", n);
	}
	printf("\n");
	return 0;
}

2.2 while循环和do-while循环

例题2.2 3n + 1问题(任何大于1的数n,若为奇数,变为3n + 1,若为偶数变为n/2,经过若干这样的变换后,最终变为1,求变换次数,其中n <= 10^9)
算法分析:若用int n来存放输入的数,当输入的n = 987654321时,对于n = 3n + 1;将导致n溢出int型范围值(-2147483648~2147483647),使得n值为负值,不满足n >1,故需要用long long型(-2^63~2^63-1)存放已输入的n值。
由于不知道循环次数,本题需要用到while循环,当while(n > 1)不满足时跳出循环。
#include <stdio.h>
int main()
{
	int n;
	scanf("%d", &n);
	long long n1 = n;
	int count = 0;
	while(n1 > 1)
	{
		if(n1 % 2 == 0) n1 = n1/2;
		else n1 = 3*n1 + 1;
		count++;
	}
	printf("%d\n", count);
	return 0;
}
例题2.3 近似计算(计算π/4 = 1 - 1/3 + 1/5 - 1/7 +......,直到最后一项小于10^-6)
算法分析:由于最后一项需要计算在内,故用do-while循环。
#include <stdio.h>
int main()
{
	double pi = 0.0, temp, t = 1.0;
	int i = 1;
	do{
		temp = 1.0/i;   //注意分子为1.0,此时才能进行浮点数运算,若写成1,temp的值将为1或者0,得不到正确结果
		pi += temp*t;  
		t = -t;
		i += 2;
	}while(temp >= 1e-6);   //注意10^-6的写法
	printf("%.6f\n", pi);
	return 0;
}

2.3 循环的代价

例题2.4 阶乘之和的末6位(计算s = 1! + 2! + 3! + ......n!,其中n<=10^6)
算法分析:当计算第n项n!时,若n比较大,会超过int的范围。需要用到一点数学知识:要计算只包含加法,减法和乘法的整数表达式除以正整数n的余数,可以在每步计算之后对n取余,结果不变。
#include <stdio.h>
#include <time.h>
int main()
{
	const int MOD = 1000000;
	int n;
	scanf("%d", &n);
	double t1 = (double)clock() / CLOCKS_PER_SEC;  //循环体开始执行前时间,防止将键盘输入时间计算在内
	int S = 0;
	for(int i = 1; i <= n; i++){
		int factorial = 1;
		for(int j = 1; j <= i; j++){
			factorial = factorial * j % MOD;  
		}
		S = (S + factorial) % MOD;            //没进行一次乘法或者加法,都要对MOD取余,使结果尽可能小
	}
	printf("%d\n", S);
	printf("Time used = %.6f\n", (double)clock() / CLOCKS_PER_SEC - t1); //使用clock()和CLOCKS_PER_SEC来计算时间差(秒)
	return 0;
}
可以将n设置160,1600,6400……来进行多组测试,可知程序的运行时间大致与n的平方成正比。从40开始,答案始终不变,这是因为25!的末尾后面有6个0,故可修改原程序,在输入n后,加一条语句if(n > 25) n = 25;

2.4 算法竞赛的输入输出框架

例题2.5 数据统计(输入一些整数,求出它们的最小值,最大值和平均值)
算法一:设置一个INF和-INF,每输入一个x,就对其进行比较,然后累加。
#include <stdio.h>
#include <cstdlib>
int main()
{
	int x, count = 0;
	double s = 0;
	int min = INT_MAX, max = INT_MIN;  //也可以先输入一个x,然后令max = min = x;
	while(scanf("%d", &x) != EOF){  //scanf("%d", &x) == EOF
		if(x > max) max = x;
		if(x < min) min = x;
		s += x;
		count++;
	}
	printf("%d %d %f\n", min, max, s/count);
	return 0;
}
scanf遇到回车键时将数据从缓冲区送到程序,例如输入5, 8, 9, 14, 16,按回车键,再按crtl + z键结束输入后,程序运行最终结果
算法二:一种好的方法是用文件——把输入数据保存在文件中,输出数据也保存在文件中。

例题2.6 数据统计ii(与上例相同,但告诉了n值,以及要求测试多组数据,输入n = 0时结束输入)
算法分析:若此处用自己设置的max和min,要注意上组数据对下组数组的影响,故应当将max和min在循环体内重新赋初值。
#include <stdio.h>
#define INF 100000000
int main()
{
	int x, n, kase = 0, min, max;
	while(scanf("%d", &n) == 1 && n)  //如果程序鲁棒性强,有时能在数据有瑕疵的情况下仍然给出正确的结果
	{
		max = -INF;
		min = INF;
		int s = 0;
		for(int i = 0; i < n; i++){
			scanf("%d", &x);
			s += x;
			if(x < min) min = x;
			if(x > max) max = x;
		}
		printf("Case %d: %d %d %.3f\n", ++kase, min, max, (double)s/n);
	}
	return 0;
}

2.5 注解和习题

习题2.4:子序列的和(输入两正整数n < m < 10^5,输出1/n^2 + 1/(n+1)^2 + ……+1/m^2,保留5位小数,输入包含多组数据,n=m=0时结束输入)
算法分析:需要注意的就是每一项都应该转换为浮点数,然后累加。用C++实现如下:
<span style="font-size:10px;">#include <iostream>
#include <iomanip>
using namespace std;
int main()
{
	double n,m;
	int kase=0;
	while(cin>>n>>m)
	{
		if(n==0&&m==0) break;
		cout<<"Case "<<++kase<<": ";
		double sum=0;
		for(int i=n;i<=m;i++)
		{
			double temp=1.0/(i*i);
			sum+=temp;
		}
		cout<<setiosflags(ios::fixed)<<setprecision(5)<<sum<<endl;
	}
	return 0;
}</span>
习题2.5:分数化小数(输入正整数a,b,c,输出a/b的小数形式,精确到小数点后c位,a,b<=10^6, c<=100,输入包含多组数据,a=b=c=0时结束输入)
算法分析:
<span style="font-size:10px;">#include <iostream>
using namespace std;
int ans1[110],ans2[110];
int main()
{
	int a,b,c;
	int kase=0;
	while(cin>>a>>b>>c)
	{
		if(a==0&&b==0&&c==0) break;
		cout<<"Case "<<++kase<<": ";
		ans1[0]=a;
		ans2[0]=a/b;
		for(int i=1;i<=c+1;i++)
		{
			ans1[i]=ans1[i-1]%b*10;  //存放每次的除数
			ans2[i]=ans1[i]/b;       //存放每次的商
		}
		if(ans2[c+1]>=5)             //进位
		{
			ans2[c]=ans2[c]+1;
			int i=c;
			while(ans2[i]==10)
			{
				if(--i>0)
					ans2[i]=ans2[i]+1;
				else break;
				ans2[i]=0;
			}
			if(ans2[1]==10) {ans2[1]=0; ans2[0]++;}   //修改整数位
		}
		cout<<ans2[0]<<".";
		for(int i=1;i<=c;i++)
			cout<<ans2[i];
		cout<<"\n";
	}
	return 0;
}</span>
习题2.6:排列(用1~9,组成3个三位数abc,def和ghi,每个数字恰好使用一次,使abc:def:ghi = 1:2:3,按格式“abc def ghi”输出所有解,每行一个)
算法分析:
#include <stdio.h>  
int  main(void)  
{  
    int i, j, k;  
    int i1, i2, i3;  
    int j1, j2, j3;  
    int k1, k2, k3;  
    int c, count;  
    for (i = 123; i <= 329; i++) {  
        j = i*2;  
        k = i*3;  
        i1 = i/100, i2 = i/10%10; i3 = i%10;  
        j1 = j/100, j2 = j/10%10; j3 = j%10;  
        k1 = k/100, k2 = k/10%10; k3 = k%10;          
        for (c = 1; c != 10; c++) {  
            count = 0;  
            if (c == i1)  
                count++;  
            if (c == i2)  
                count++;  
            if (c == i3)  
                count++;  
            if (c == j1)  
                count++;  
            if (c == j2)  
                count++;  
            if (c == j3)  
                count++;  
            if (c == k1)  
                count++;  
            if (c == k2)  
                count++;  
            if (c == k3)  
                count++;  
            if (count > 1)  
                break;  
        }  
        if (c == 10 && i2 != 0 && i3 != 0 && j2 != 0 && j3 != 0 && k2 != 0 && k3 != 0)  
            printf("%d %d %d\n", i, j, k);  
    }         
    return 0;  
}  

































评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值