update dir

This commit is contained in:
light-city 2019-07-26 15:13:47 +08:00
parent 5818287aa3
commit bf8b7ef7ff
15 changed files with 246 additions and 3 deletions

View File

@ -17,6 +17,7 @@
- [x] [volatile那些事](./volatile)
- [x] [assert那些事](./assert)
- [x] [位域那些事](./bit)
- [x] [extern那些事](./extern)
## 关于作者:
个人公众号:

View File

@ -1,5 +1,13 @@
# assert那些事
## 关于作者:
个人公众号:
![](../img/wechat.jpg)
## 1.第一个断言案例
断言,**是宏,而非函数**。assert 宏的原型定义在 <assert.h>C<cassert>C++)中,其作用是如果它的条件返回错误,则终止程序执行。可以通过定义 NDEBUG 来关闭 assert但是需要在源代码的开头include <assert.h> 之前。

View File

@ -1,8 +1,8 @@
## 关于作者:
个人公众号:
[TOC]
![](../img/wechat.jpg)
## Bit field 是什么?

193
extern/README.md vendored Normal file
View File

@ -0,0 +1,193 @@
# extern "C"
## 关于作者:
个人公众号:
![](../img/wechat.jpg)
## 1.C++与C编译区别
在C++中常在头文件见到extern "C"修饰函数,那有什么作用呢? 是用于C++链接在C语言模块中定义的函数。
C++虽然兼容C但C++文件中函数编译后生成的符号与C语言生成的不同。因为C++支持函数重载C++函数编译后生成的符时带有函数参数类型的信息而C则没有。
例如`int add(int a, int b)`函数经过C++编译器生成.o文件后`add`会变成形如`add_int_int`之类的, 而C的话则会是形如`_add`, 就是说相同的函数在C和C++中,编译后生成的符号不同。
这就导致一个问题如果C++中使用C语言实现的函数在编译链接的时候会出错提示找不到对应的符号。此时`extern "C"`就起作用了:告诉链接器去寻找`_add`这类的C语言符号而不是经过C++修饰的符号。
## 2.C++调用C函数
C++调用C函数的例子: 引用C的头文件时需要加`extern "C"`
```c++
//add.h
#ifndef ADD_H
#define ADD_H
int add(int x,int y);
#endif
//add.c
#include "add.h"
int add(int x,int y) {
return x+y;
}
//add.cpp
#include <iostream>
#include "add.h"
using namespace std;
int main() {
add(2,3);
return 0;
}
```
编译:
```
//Generate add.o file
gcc -c add.c
```
链接:
```
g++ add.cpp add.o -o main
```
没有添加extern "C" 报错:
```c++
> g++ add.cpp add.o -o main
add.o在函数main
add.cpp:(.text+0x0): `main'被多次定义
/tmp/ccH65yQF.o:add.cpp:(.text+0x0):第一次在此定义
/tmp/ccH65yQF.o在函数main
add.cpp:(.text+0xf)add(int, int)’未定义的引用
add.o在函数main
add.cpp:(.text+0xf)add(int, int)’未定义的引用
collect2: error: ld returned 1 exit status
```
添加extern "C"后:
`add.cpp`
```c++
#include <iostream>
using namespace std;
extern "C" {
#include "add.h"
}
int main() {
add(2,3);
return 0;
}
```
编译的时候一定要注意先通过gcc生成中间文件add.o。
```
gcc -c add.c
```
然后编译:
```
g++ add.cpp add.o -o main
```
上述案例源代码见:
- [add.h](extern_c++/add.h)
- [add.c](extern_c++/add.c)
- [add.cpp](extern_c++/add.cpp)
## 2.C中调用C++函数
`extern "C"`在C中是语法错误需要放在C++头文件中。
```c
// add.h
#ifndef ADD_H
#define ADD_H
extern "C" {
int add(int x,int y);
}
#endif
// add.cpp
#include "add.h"
int add(int x,int y) {
return x+y;
}
// add.c
extern int add(int x,int y);
int main() {
add(2,3);
return 0;
}
```
编译:
```c
g++ -c add.cpp
```
链接:
```
gcc add.c add.o -o main
```
上述案例源代码见:
- [add.h](extern_c/add.h)
- [add.c](extern_c/add.c)
- [add.cpp](extern_c/add.cpp)
综上总结出使用方法在C语言的头文件中对其外部函数只能指定为extern类型C语言中不支持extern "C"声明,在.c文件中包含了extern "C"时会出现编译语法错误。所以使用extern "C"全部都放在于cpp程序相关文件或其头文件中。
总结出如下形式:
1C++调用C函数
```c++
//xx.h
extern int add(...)
//xx.c
int add(){
}
//xx.cpp
extern "C" {
#include "xx.h"
}
```
2C调用C++函数
```c
//xx.h
extern "C"{
int add();
}
//xx.cpp
int add(){
}
//xx.c
extern int add();
```

5
extern/extern_c++/add.c vendored Normal file
View File

@ -0,0 +1,5 @@
#include "add.h"
int add(int x,int y) {
return x+y;
}

9
extern/extern_c++/add.cpp vendored Normal file
View File

@ -0,0 +1,9 @@
#include <iostream>
using namespace std;
extern "C" {
#include "add.h"
}
int main() {
add(2,3);
return 0;
}

4
extern/extern_c++/add.h vendored Normal file
View File

@ -0,0 +1,4 @@
#ifndef ADD_H
#define ADD_H
extern int add(int x,int y);
#endif

BIN
extern/extern_c++/add.o vendored Normal file

Binary file not shown.

BIN
extern/extern_c++/main vendored Executable file

Binary file not shown.

5
extern/extern_c/add.c vendored Normal file
View File

@ -0,0 +1,5 @@
extern int add(int x,int y);
int main() {
add(2,3);
return 0;
}

5
extern/extern_c/add.cpp vendored Normal file
View File

@ -0,0 +1,5 @@
#include "add.h"
int add(int x,int y) {
return x+y;
}

6
extern/extern_c/add.h vendored Normal file
View File

@ -0,0 +1,6 @@
#ifndef ADD_H
#define ADD_H
extern "C" {
int add(int x,int y);
}
#endif

BIN
extern/extern_c/add.o vendored Normal file

Binary file not shown.

BIN
extern/extern_c/main vendored Executable file

Binary file not shown.

View File

@ -1,4 +1,11 @@
## 关于作者:
个人公众号:
![](../img/wechat.jpg)
## 1.volatile
`volatile` 修饰的变量,在对其进行读写操作时,会引发一些**可观测的副作用**。而这些可观测的副作用,是由**程序之外的因素决定的**。
## 2.volatile应用