两个简单的C语言问题
C语言新手,问两个简单的问题,高手看看.
1. #include<stdio.h>
#define M 3
#defme N M+1
#define NN N*N/2
main() .
{printf("%d\n",NN);}
答案是6.为何N是4,NN为6呢?
2.main (){ int n=1;
printf("%d %d %d\n",n,n++,n--);
}
为什么输出是1 0 1?N首先是1,然后应该打印1,++后为2,最后应该打印2,再--变成1,不知道哪里错了?
[解决办法]
第一个是宏替换
M+1*M+1/2=6.5输出整数部分6
#include<stdio.h>
#define M 3
#defme N M+1
#define NN (N)*(N)/2
main() .
{printf("%d\n",NN);}
应该这样
[解决办法]
第一个问题:
#define NN N*N/2
你这个宏定没有加保护,预编译进行替换后变成如下这样子
3+1*3+1/2 = 6
所以NN = 6;
你要改成这样子
#define NN (N)*(N)/2
第二个问题:
2.main (){ int n=1;
printf("%d %d %d\n",n,n++,n--);
}
为什么输出是1 0 1?N首先是1,然后应该打印1,++后为2,最后应该打印2,再--变成1,不知道哪里错了?
答:
这个是编译器有关
因为printf()是个函数。这三个数是同时执行的,他首先算N-- 然后再算N++ 其实这个结果对编译器来说是正确的,便对你的结果是错语的。如果你想要你的结果,可以改成如下:这就可以输出 1 1 2
root@karl-u:/# cat main.c
int main ()
{
int n=1;
//printf("%d %d %d\n",n,n++,n--);
printf("%d \n",n);
printf("%d \n",n++);
printf("%d \n",n--);
return 0;
}
root@karl-u:/# ./a.out
1
1
2
[解决办法]
#与##在宏定义中的--宏展开
#include <stdio.h>
#define f(a,b) a##b
#define g(a) #a
#define h(a) g(a)
int main()
{
printf("%s\n", h(f(1,2))); // 12
printf("%s\n", g(f(1,2))); // f(1,2)
return 0;
}
宏展开时:
如果宏定义以#开头,不展开参数,直接替换。
故g(f(1,2))--->#f(1,2)--->"f(1,2)";
如果宏定义不以#开头,展开参数,直接替换,由外层向里层,如果碰到的是#开头的宏,不继续往里层展开,往外层展开。
由外层向里层,如果碰到的是以非#开头的宏,继续往里层走,直至最里层,开始一层层往外层展开。
故h(f(1,2))--->h(12)--->g(12)---->#12----->"12"。
PS:
##在宏中定义,是字符连接符
如a##b##c 等同于 "abc"
#在宏开头出现,是表示宏展开的方式不同
#a 等同于"a"
#abc 等同于 "abc"
复杂的:
#include <stdio.h>
#define f(a,b) a##b
#define g(a) #a
#define h(a) g(a)
int main()
{
char a = 'a';
cout<<g(a)<<endl; // a
cout<<g(g(a))<<endl; // a
printf("%s\n", h(f(1,2))); // 12
printf("%s\n", g(f(1,2))); // f(1,2)
printf("%s\n", g(h(f(1,2)))); // h(f(1,2))
printf("%s\n", h(g(f(1,2)))); // "f(1,2)"
printf("%s\n", h(h(f(1,2)))); // "12"
system("pause");
return 0;
}
预处理后的:(在编译选项中添加/EP /P后编译生成的.i文件)
int main()
{
char a = 'a';
cout<<"a"<<endl;
cout<<"g(a)"<<endl;
printf("%s\n", "12");
printf("%s\n", "f(1,2)");
printf("%s\n", "h(f(1,2))");
printf("%s\n", "\"f(1,2)\"");
printf("%s\n", "\"12\"");
system("pause");
return 0;
}
---------------------------------------------------
宏解析
1. ##操作符
##操作符它的作用是在替代表中将其前后的参数连接成为一个预处理符号,它不能出现于宏替代表的开端和末尾。
例:
#define concat(s,t) s##t
#define AAA ABC
concat(A, AA)
将被替换成
ABC
2. 重新扫描和替换
在替换列表中的所有参数替换过之后,预处理器将对结果token序列重新扫描以便对其中的宏再次替换。
当正在替换的宏在其替换列表中发现自身时,就不再对其进行替换。今儿,在任何正在嵌套替换的宏的替换过程中遇到正被替换的宏就对其不再进行替换(防止递归)。
例:
#define ROOT AAA CCC
#define AAA ROOT
ROOT
将被替换成
ROOT CCC
[解决办法]
第二个问题和入栈顺序有关,编译器按从右到左入栈,即 n--、n++、n,结果就是1、0、1了。可以写成
int n=1;
printf("%d %d %d %d %d %d \n", n, 5, n++, 6, n--, 7);
观察一下567的入栈顺序
int n=1;00411BBE mov dword ptr [n],1 printf("%d %d %d %d %d %d \n", n, 5, n++, 6, n--, 7);00411BC5 mov eax,dword ptr [n] 00411BC8 mov dword ptr [ebp-0DCh],eax 00411BCE mov ecx,dword ptr [n] 00411BD1 sub ecx,1 00411BD4 mov dword ptr [n],ecx 00411BD7 mov edx,dword ptr [n] 00411BDA mov dword ptr [ebp-0E0h],edx 00411BE0 mov eax,dword ptr [n] 00411BE3 add eax,1 00411BE6 mov dword ptr [n],eax 00411BE9 mov esi,esp 00411BEB push 7 00411BED mov ecx,dword ptr [ebp-0DCh] 00411BF3 push ecx 00411BF4 push 6 00411BF6 mov edx,dword ptr [ebp-0E0h] 00411BFC push edx 00411BFD push 5 00411BFF mov eax,dword ptr [n] 00411C02 push eax 00411C03 push offset string "%d %d %d %d %d %d \n" (415B68h) 00411C08 call dword ptr [__imp__printf (4182CCh)] 00411C0E add esp,1Ch 00411C11 cmp esi,esp 00411C13 call @ILT+370(__RTC_CheckEsp) (411177h)