1、i++,i++;的纠结

例:

#include <stdio.h>


int main(void)

{

    int i = 5;

    printf("%d %d", i++, i++);

    return 0;

}


解析:这个问题,在实际编程应该是没有人会这么写的,可还是有很多初学者纠结于此。在编程这个领域里,很多时候,实践是最好的老师。对于代码有疑问,那么上机敲一遍,编译运行一遍是很好的方法。但是,在i++,i++这个方面,即使编译运行了这个程序,也不一定会有正确的结果。用VC,TC,gcc编译运行后的结果不一定相同。或者从某个角度来说,这里结果的正确与否其实并不重要了。

printf("%d %d", i++, i++);这一个语句,其中的i++,i++是未指定行为。即C语言的标准并没有指定这运算是以何种顺序进行的。如果用gcc编译加上-Wall选项,那么会有warning:

gcc 2_1.c -o 2_1 -g -Wall

2_1.c: 在函数‘main’中:

2_1.c:6:24: 警告:‘i’上的运算结果可能是未定义的

因此,要解决这个问题,最好的方法就是不要在程序中写这样的代码。

解决方案:见上。


2、60<=grade<=70

例:

...

if (60 <= grade <= 70)

   printf("及格");

else if(70 <= grade <= 85)

     printf("良好");

...


解析:在数学中,60<=grade<=70这种表达是成立的,但是在C语言中,并没有这种表达.

解决方案:应该改写为

if ((60 <= grade) && (grade <= 70))

   printf("及格");

else ...


3、switch接受什么值?

例2


#include <stdio.h>


int main(void)

{

    double choice;

    scanf("%lf", &choice);

    switch(choice){

    case 1.0 : printf("1.0");

        break;

    case 2.0 : printf("2.0");

        break;

    default : printf("It's not 1.0 or 2.0");

    }

    return 0;

}


解析:这种问题同样是不会出现在实际的编程当中。但是一些C语言题目可能会这么出。switch()接受的是整数:整型或者字符型。所以,浮点型,字符串等类型是不被switch()接受的。


解决方案:无他……


4、== =的困惑

例:

/****************************************

Copyright(C), 1999-2011, XX Co., Ltd.

FileName:7121.c

Author:……… Version:1.0

Time:23:06 Date:Feb 13th, 2011

Description:C Primer Plus编程练习7.12 1

Function List:

1.int main(void)

2.void count(void);

3.void show_count(void);

History:

<author> <time> <date> <version> <desc>

……… 23:06 2011/2/13 1.0

****************************************/


#include<stdio.h>

#include<stdlib.h>

#define STOP '#' /*以#为输入结束*/

#define SPACE ' ' /*定义空格*/

#define ENTER '\n' /*定义回车*/


/*空格计数器,换行符计数器,其他字符计数器,用户输入字符*/

int sp_count = 0;

int n_count = 0;

int other_count = 0;

char ch;


void count(void);/*计数器函数*/

void show_count(void);/*报告计数结果*/


int main(void)

{

    puts("Please enter some letters:(# to end)");

    count();

    show_count();


    puts("Thank you for using this program made by HerBal_Tea!");

    system("pause");

    return 0;

}


void count(void)

{

    while ((ch = getchar()) != STOP)

    {

        if (ch = SPACE)

        {

            sp_count++;

            continue;

        }


        if (ch = ENTER)

        {

            n_count++;

            continue;

        }


        other_count++;

    }/*end of while ((ch = getchar()) != '#')*/

}


void show_count(void)

{

    puts("The number of \nspace enter other");

    printf("%5d%6d%6d\n", sp_count, n_count, other_count);

}

解析:话说我自己也犯了这个错。在数学中,=表示相等,而在很多编程语言中,=表示赋值,==才表示相等。思维惯性导致错误。而且这个错误C编译器既不会报错也不会警告。出错了检查起来非常难。我看了不下二十遍调试的值监视都找不到,最后还是一行一行源代码看,才知道是这个问题。在C语言四书五经中的 c traps and pitfalls 和 expert c programming都提到过这个问题。


解决方案:无他,唯细心。


5、溢出问题

例:求斐波那契数列的前n项

#include <stdio.h>


#define N 100


int main(void)

{

    int fib[N];

    int i;


    fib[0] = fib[1] = 1;

    printf("%d %d ", fib[0], fib[1]);

    for (i = 2; i < N; ++i)

    {

        fib[i] = fib[i - 1] + fib[i - 2];

        printf("%d ", fib[i]);

    }

    return 0;

}


解析:在数学的概念中,整数、小数都是无限的,但是计算机中,即使是long long int或者double类型都是有一定限度的,超过限度就会溢出。即是,假设一个指针式体重秤最大量程为120KG,那么一个体重130KG的人站上去,那么指针会指向10KG的刻度处。溢出同理。而C语言,编译器是不会对溢出进行检查或者处理的。因此在编程中,应该自己估算一下数的大小,以选用合适的数据类型来表示数据。PS:由于TC编译出的程序是16位的,所以int也是16位,很容易就会溢出。


解决方案:一方面是使用合适的数据类型,比如long ,long long 或者double。另一方面是,如果数特别大,连unsigned long long int或者连double都表示不了的时候,那么就用数组吧。不过定义数组的加减法还算好,但是定义乘除法就不是那么简单了。所以推荐新人使用第一种方法。第二种方法,可以考虑自己编个小程序实现一下。


6、四舍五入?

例:要求将输入的数按指定精度四舍五入输出

#include <stdio.h>


int main(void)

{

    double a = 158.385427;

    printf("%.2lf", a);

    return 0;

}


解析:C语言中,printf("%.2lf", num);是直接截断至小数点后两位,并非四舍五入而是类似于趋零截尾。


解决方案:如下

//四舍五入显示数字,精确由用户输入

//可辨别正负

#include<stdio.h>

#include<stdlib.h>

#include<math.h>

int main(void)

{

    int width;

    double a = 158.385427;

    puts("Enter the width");

    scanf("%d",&width);

    if(a>0.0)

    a = (int)(a*pow(10,width) + 0.5)/pow(10,width);

    else

    a = (int)(a*pow(10,width) - 0.5)/pow(10,width);

    printf("The result is %.*lf\n",width,a);

    system("pause");

    return 0;

}



7、函数的返回值

例:

#include<stdio.h>

double salary();/*计算工资税的函数*/

void main()

{

    int choice;

    char want0;

    double result;

    while(1)

    {

        printf("欢迎使用个人所得税计算器\n");

        printf("\n");

        printf("1.工资、薪金所得税计算\n");

    printf("\n");

        printf("请输入需要计算的税收项目序号:");

        scanf("%d", &choice);

        if(choice==1)

        {

            salary();

            printf("%lf", result);

        }

double salary()

{

    double sal, result;

    printf("请输入您的薪水:");

    scanf("%lf", &sal);

    if(sal<=3000)

    {

        result=sal;

    }

        if(sal>3000&&sal<=4500)

        {

            result=sal*0.05;

        }

        if(sal>4500&&sal<=7500)

        {

            result=sal*0.1-75;

        }

        if(sal>7500&&sal<=12000)

        {

            result=sal*0.2-525;

        }

        if(sal>12000&&sal<=38000)

        {

            result=sal*0.25-975;

        }

        if(sal>38000&&sal<=58000)

        {

            result=sal*0.3-2725;

        }

        if(sal>58000&&sal<=83000)

        {

            result=sal*0.35-5475;

        }

        if(sal>83000)

        {

            result=sal*0.45-13475;

        }

    return result;

}


解析:这个问题简单来说是,用户定义函数并未返回计算所得数值。往深的来说是关于存储类、链接,即变量的作用域、链接、存储时期的问题。


解决方案:

/*

 * main.c

 *

 * Created on: 2011-6-11

 * Author: ice

 */


#include<stdio.h>

double salary();/*计算工资税的函数*/

int main(void)

{

    int choice;

//    char want0;////这个want0变量没有使用

    double result;

    while(1)

    {

        printf("欢迎使用个人所得税计算器\n");

        printf("\n");

        printf("1.工资、薪金所得税计算\n");

     printf("\n");

        printf("请输入需要计算的税收项目序号:");

        scanf("%d", &choice);

        if(choice==1)

        {

            result = salary();//要有一个值接受返回值才可以

            printf("%lf", result);

        }

    }

}


double salary()

{

    double sal, result;

    printf("请输入您的薪水:");

    scanf("%lf", &sal);

    if(sal<=3000)

    {

        result=sal;

    }

        if(sal>3000&&sal<=4500)

        {

            result=sal*0.05;

        }

        if(sal>4500&&sal<=7500)

        {

            result=sal*0.1-75;

        }

        if(sal>7500&&sal<=12000)

        {

            result=sal*0.2-525;

        }

        if(sal>12000&&sal<=38000)

        {

            result=sal*0.25-975;

        }

        if(sal>38000&&sal<=58000)

        {

            result=sal*0.3-2725;

        }

        if(sal>58000&&sal<=83000)

        {

            result=sal*0.35-5475;

        }

        if(sal>83000)

        {

            result=sal*0.45-13475;

        }

    return result;

}