Skip to content

Commit 3a495d7

Browse files
committed
update dir
1 parent 09c1025 commit 3a495d7

File tree

6 files changed

+395
-0
lines changed

6 files changed

+395
-0
lines changed

Diff for: README.md

+1
Original file line numberDiff line numberDiff line change
@@ -29,6 +29,7 @@
2929
- [x] [enum那些事](./enum)
3030
- [x] [decltype那些事](./decltype)
3131
- [x] [引用与指针那些事](./pointer_refer)
32+
- [x] [宏那些事](./macro)
3233

3334
代码运行:
3435
全部在linux下用vim编写,使用gcc/g++调试!全部可正常运行!

Diff for: macro/README.md

+232
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,232 @@
1+
# 宏那些事
2+
3+
## 关于作者:
4+
5+
个人公众号:
6+
7+
![](../img/wechat.jpg)
8+
9+
## 1.宏中包含特殊符号
10+
11+
分为几种:`#``##``\`
12+
13+
### 1.1 字符串化操作符(#)
14+
15+
**在一个宏中的参数前面使用一个#,预处理器会把这个参数转换为一个字符数组**,换言之就是:**#是“字符串化”的意思,出现在宏定义中的#是把跟在后面的参数转换成一个字符串**
16+
17+
**注意:其只能用于有传入参数的宏定义中,且必须置于宏定义体中的参数名前。**
18+
19+
例如:
20+
21+
```c++
22+
#define exp(s) printf("test s is:%s\n",s)
23+
#define exp1(s) printf("test s is:%s\n",#s)
24+
#define exp2(s) #s
25+
int main() {
26+
exp("hello");
27+
exp1(hello);
28+
29+
string str = exp2( bac );
30+
cout<<str<<" "<<str.size()<<endl;
31+
/**
32+
* 忽略传入参数名前面和后面的空格。
33+
*/
34+
string str1 = exp2( asda bac );
35+
/**
36+
* 当传入参数名间存在空格时,编译器将会自动连接各个子字符串,
37+
* 用每个子字符串之间以一个空格连接,忽略剩余空格。
38+
*/
39+
cout<<str1<<" "<<str1.size()<<endl;
40+
return 0;
41+
}
42+
```
43+
44+
上述代码给出了基本的使用与空格处理规则,空格处理规则如下:
45+
46+
- 忽略传入参数名前面和后面的空格。
47+
48+
```c++
49+
string str = exp2( bac );
50+
cout<<str<<" "<<str.size()<<endl;
51+
```
52+
53+
输出:
54+
55+
```
56+
bac 3
57+
```
58+
59+
- 当传入参数名间存在空格时,编译器将会自动连接各个子字符串,用每个子字符串之间以一个空格连接,忽略剩余空格。
60+
61+
```c++
62+
string str1 = exp2( asda bac );
63+
cout<<str1<<" "<<str1.size()<<endl;
64+
```
65+
66+
输出:
67+
68+
```
69+
asda bac 8
70+
```
71+
72+
### 1.2 符号连接操作符(##)
73+
74+
**“##”是一种分隔连接方式,它的作用是先分隔,然后进行强制连接。将宏定义的多个形参转换成一个实际参数名。**
75+
76+
注意事项:
77+
78+
**(1)当用##连接形参时,##前后的空格可有可无。**
79+
80+
**(2)连接后的实际参数名,必须为实际存在的参数名或是编译器已知的宏定义。**
81+
82+
**(3)如果##后的参数本身也是一个宏的话,##会阻止这个宏的展开。**
83+
84+
示例:
85+
86+
```c++
87+
88+
#define expA(s) printf("前缀加上后的字符串为:%s\n",gc_##s) //gc_s必须存在
89+
// 注意事项2
90+
#define expB(s) printf("前缀加上后的字符串为:%s\n",gc_ ## s) //gc_s必须存在
91+
// 注意事项1
92+
#define gc_hello1 "I am gc_hello1"
93+
int main() {
94+
// 注意事项1
95+
const char * gc_hello = "I am gc_hello";
96+
expA(hello);
97+
expB(hello1);
98+
}
99+
```
100+
101+
### 1.3 续行操作符(\)
102+
103+
**当定义的宏不能用一行表达完整时,可以用”\”表示下一行继续此宏的定义。**
104+
105+
**注意 \ 前留空格。**
106+
107+
```c++
108+
#define MAX(a,b) ((a)>(b) ? (a) \
109+
:(b))
110+
int main() {
111+
int max_val = MAX(3,6);
112+
cout<<max_val<<endl;
113+
}
114+
```
115+
116+
上述代码见:[sig_examp.cpp](sig_examp.cpp)
117+
118+
## 2.do{...}while(0)的使用
119+
120+
### 2.1 避免语义曲解
121+
122+
例如:
123+
124+
```
125+
#define fun() f1();f2();
126+
if(a>0)
127+
fun()
128+
```
129+
130+
这个宏被展开后就是:
131+
132+
```
133+
if(a>0)
134+
f1();
135+
f2();
136+
```
137+
138+
本意是a>0执行f1 f2,而实际是f2每次都会执行,所以就错误了。
139+
140+
为了解决这种问题,在写代码的时候,通常可以采用`{}`块。
141+
142+
如:
143+
144+
```c++
145+
#define fun() {f1();f2();}
146+
if(a>0)
147+
fun();
148+
// 宏展开
149+
if(a>0)
150+
{
151+
f1();
152+
f2();
153+
};
154+
```
155+
156+
但是会发现上述宏展开后多了一个分号,实际语法不太对。(虽然编译运行没问题,正常没分号)。
157+
158+
### 2.2避免使用goto控制流
159+
160+
在一些函数中,我们可能需要在return语句之前做一些清理工作,比如释放在函数开始处由malloc申请的内存空间,使用goto总是一种简单的方法:
161+
162+
```c++
163+
int f() {
164+
int *p = (int *)malloc(sizeof(int));
165+
*p = 10;
166+
cout<<*p<<endl;
167+
#ifndef DEBUG
168+
int error=1;
169+
#endif
170+
if(error)
171+
goto END;
172+
// dosomething
173+
END:
174+
cout<<"free"<<endl;
175+
free(p);
176+
return 0;
177+
}
178+
```
179+
180+
但由于goto不符合软件工程的结构化,而且有可能使得代码难懂,所以很多人都不倡导使用,这个时候我们可以使用do{...}while(0)来做同样的事情:
181+
182+
```c++
183+
int ff() {
184+
int *p = (int *)malloc(sizeof(int));
185+
*p = 10;
186+
cout<<*p<<endl;
187+
do{
188+
#ifndef DEBUG
189+
int error=1;
190+
#endif
191+
if(error)
192+
break;
193+
//dosomething
194+
}while(0);
195+
cout<<"free"<<endl;
196+
free(p);
197+
return 0;
198+
}
199+
```
200+
201+
这里将函数主体部分使用do{...}while(0)包含起来,使用break来代替goto,后续的清理工作在while之后,现在既能达到同样的效果,而且代码的可读性、可维护性都要比上面的goto代码好的多了。
202+
203+
### 2.3 避免由宏引起的警告
204+
205+
内核中由于不同架构的限制,很多时候会用到空宏,。在编译的时候,这些空宏会给出warning,为了避免这样的warning,我们可以使用do{...}while(0)来定义空宏:
206+
207+
```
208+
#define EMPTYMICRO do{}while(0)
209+
```
210+
211+
### 2.4 **定义单一的函数块来完成复杂的操作**
212+
213+
如果你有一个复杂的函数,变量很多,而且你不想要增加新的函数,可以使用do{...}while(0),将你的代码写在里面,里面可以定义变量而不用考虑变量名会同函数之前或者之后的重复。
214+
这种情况应该是指一个变量多处使用(但每处的意义还不同),我们可以在每个do-while中缩小作用域,比如:
215+
216+
```c++
217+
int fc()
218+
{
219+
int k1 = 10;
220+
cout<<k1<<endl;
221+
do{
222+
int k1 = 100;
223+
cout<<k1<<endl;
224+
}while(0);
225+
cout<<k1<<endl;
226+
}
227+
```
228+
229+
上述代码见:[do_while.cpp](do_while.cpp)
230+
231+
学习文章:<https://www.cnblogs.com/lizhenghn/p/3674430.html>
232+

Diff for: macro/do_while

9.24 KB
Binary file not shown.

Diff for: macro/do_while.cpp

+76
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,76 @@
1+
#include <iostream>
2+
#include <malloc.h>
3+
4+
using namespace std;
5+
#define f1() cout<<"f1()"<<endl;
6+
#define f2() cout<<"f2()"<<endl;
7+
8+
#define fun() {f1();f2();}
9+
#define fun1() \
10+
do{ \
11+
f1();\
12+
f2();\
13+
}while(0)
14+
15+
int f() {
16+
int *p = (int *)malloc(sizeof(int));
17+
*p = 10;
18+
cout<<*p<<endl;
19+
#ifndef DEBUG
20+
int error=1;
21+
#endif
22+
if(error)
23+
goto END;
24+
25+
// dosomething
26+
END:
27+
cout<<"free"<<endl;
28+
free(p);
29+
return 0;
30+
}
31+
32+
int ff() {
33+
34+
int *p = (int *)malloc(sizeof(int));
35+
*p = 10;
36+
cout<<*p<<endl;
37+
38+
do{
39+
#ifndef DEBUG
40+
int error=1;
41+
#endif
42+
if(error)
43+
break;
44+
//dosomething
45+
}while(0);
46+
47+
cout<<"free"<<endl;
48+
free(p);
49+
return 0;
50+
}
51+
52+
53+
int fc()
54+
{
55+
int k1 = 10;
56+
cout<<k1<<endl;
57+
do{
58+
int k1 = 100;
59+
cout<<k1<<endl;
60+
}while(0);
61+
cout<<k1<<endl;
62+
}
63+
64+
int main() {
65+
66+
if(1>0)
67+
fun();
68+
69+
if(2>0)
70+
fun1();
71+
72+
f();
73+
ff();
74+
fc();
75+
return 0;
76+
}

Diff for: macro/sig_examp

14 KB
Binary file not shown.

0 commit comments

Comments
 (0)