有这么一道面试题:
What will the be the output of the following C code?
#define product(x) (x*x)
int main(int argc, char* argv[])
{
int i=3,j,k;
j = product(i++);//i++*i++
printf("i = %d\n",i);
k = product(++i);
printf("i = %d,j = %d,k = %d\n",i,j,k);
return 0;
}
解析:producti(i++)=i++*i++;i=3,所以j=9,此时i已经累积为5,producti(++i)要求i先++,所以此时i的值是7,所以i = 7,j = 9,k = 49
在这道题中运用了宏定义,那么如果不使用宏定义,而用函数代替的结果怎么样呢?
int product(int x);
int main(int argc, char* argv[])
{
int i=3,j,k;
j = product(i++);//i++*i++
printf("i = %d\n",i);
k = product(++i);
printf("i = %d,j = %d,k = %d\n",i,j,k);
return 0;
}
int product(int x){
return x*x;
}
结果是
i = 4
i = 5,j = 9,k = 25
i = 5,j = 9,k = 25
由此可以看出宏定义的区别在哪里吧?
宏定义是C提供的三种预处理功能的其中一种,这三种预处理包括:宏定义、文件包含、条件编译
现在附上宏定义的使用方法:
使用宏时,用圆括号括住每个参数,并括住宏的整体定义;一般来说,在宏中不要使用自增或自减运算符。
[1]最值
#define MIN(x, y) (((x) < (y)) ? (x) : (y))
求两者间的较小者
#define MAX(x, y) (((x) > (y)) ? (x) : (y))
求两者间的较大者
#define MAX3(x,y,z) MAX(MAX(x,y),z)
求三者间的较大者
[2]移位
#define SHIFT(value, n) (((n) > 0) ? ((value) << (n))
: ((value) >> -(n)))
对value进行移位操作,n > 0左移位,n < 0右移位
[3]字符
#define IS_LOWER_CASE(x) (((x)>=’a’) && ((x)<=’z’))
判断小写
#define IS_UPPER_CASE(x) (((x)>=’A’) && ((x)<=’Z’))
判断大写
#define IS_ALPHABETIC(c) (IS_LOWER_CASE (c) || IS_UPPER_CASE (c))
判断字母
#define IS_DIGIT(x) (((x)>=’0’) && ((x)<=’9’))
判断数字
#define IS_SPECIAL(c) (!IS_DIGIT (c) && !IS_ALPHABETIC (c))
判断特殊字符
[4]取绝对值
#define ABSOLUTE_VALUE(x) (((x) < 0) ? -(x) : (x))
[5]#运算符
#define dprint(expr) printf(#expr " = %.1fn",expr)
#参数:将输出带引号的参数值;
dprint(1.0/2.0);将输出:1.0/2.0=0.5;
[6]##运算符
#define xprint(front,back) front ## back
xprint(x,1);将输出:x1
[7]判断闰年
#define LEAP_YEAR(x) (((x) % 4 == 0 && (x) % 100 != 0) || ((x) % 400 == 0))
[8]整数val取第n位的bit
#define BIT(VAL,N) (((VAL) >> (N)) & 1 ? 1 : 0
[1]最值
#define MIN(x, y) (((x) < (y)) ? (x) : (y))
求两者间的较小者
#define MAX(x, y) (((x) > (y)) ? (x) : (y))
求两者间的较大者
#define MAX3(x,y,z) MAX(MAX(x,y),z)
求三者间的较大者
[2]移位
#define SHIFT(value, n) (((n) > 0) ? ((value) << (n))
: ((value) >> -(n)))
对value进行移位操作,n > 0左移位,n < 0右移位
[3]字符
#define IS_LOWER_CASE(x) (((x)>=’a’) && ((x)<=’z’))
判断小写
#define IS_UPPER_CASE(x) (((x)>=’A’) && ((x)<=’Z’))
判断大写
#define IS_ALPHABETIC(c) (IS_LOWER_CASE (c) || IS_UPPER_CASE (c))
判断字母
#define IS_DIGIT(x) (((x)>=’0’) && ((x)<=’9’))
判断数字
#define IS_SPECIAL(c) (!IS_DIGIT (c) && !IS_ALPHABETIC (c))
判断特殊字符
[4]取绝对值
#define ABSOLUTE_VALUE(x) (((x) < 0) ? -(x) : (x))
[5]#运算符
#define dprint(expr) printf(#expr " = %.1fn",expr)
#参数:将输出带引号的参数值;
dprint(1.0/2.0);将输出:1.0/2.0=0.5;
[6]##运算符
#define xprint(front,back) front ## back
xprint(x,1);将输出:x1
[7]判断闰年
#define LEAP_YEAR(x) (((x) % 4 == 0 && (x) % 100 != 0) || ((x) % 400 == 0))
[8]整数val取第n位的bit
#define BIT(VAL,N) (((VAL) >> (N)) & 1 ? 1 : 0
预处理(预编译)工作也叫做宏展开:将宏名替换为字符串。
掌握"宏"概念的关键是“换”。一切以换为前提、做任何事情之前先要换,准确理解之前就要“换”。
即在对相关命令或语句的含义和功能作具体分析之前就要换:
如此可以得出,题目中为什么会有如此的结果了吧。