C99
C语言的发展历史大致上分为三个阶段:Old Style C、C89和C99。Ken Thompson和Dennis Ritchie发明C语言时有很多语法和现在并不一样,但为了向后兼容性(Backward Compatibility), 这些语法仍然在C89和C99中保留下来了。C89是最早的C语言规范,1990年由ANSI(美国国家标准委员会)推出ANSI版本,后来被接纳为ISO国际标准(ISO/IEC 9899:1990),因而有时也称为C90。C89是目前最广泛采用的C语言标准,大多数编译器都完全支持C89。C99标准(ISO/IEC 9899:1999)是在1999年推出的,加入了许多新的特性,但目前仍没有得到广泛支持。
-
inline (内联)关键字 该函数在代码内进行内联扩展,当代码执行时没有函数的进栈与退栈,函数执行速度加快。 -
新增数据类型
- 增加了用来定义
bool 、true 以及false 宏的头文件<stdbool.h> - 引进了
long long int 和unsigned long long int -
可变长数组(VLA) 声明数组时,数组的维数可以由任一有效的整型表达式确定,包括只在运行时才能确定其值的表达式,这类数组就叫做可变长数组。只有局部数组才可以是变长的,且可变长数组的维数在数组生存期内不变。 -
预处理程序的修改
-
具有可变数目的参数的宏 #define report(test, ...) ((test)?puts(#test):printf(__VA_ARGS__))
report(x>y, “x is %d but y is %d”, x, y);
((x>y)?puts(“x>y”):printf(“x is %d but y is %d”, x, y))
-
内部编译指令 STDC FP_CONTRACT ON/OFF/DEFAULT
若为ON,浮点表达式被当做基于硬件方式处理的独立单元
STDC FEVN_ACCESS ON/OFF/DEFAULT
告诉编译程序可以访问浮点环境,默认值是定义的工具.
STDC CX_LIMITED_RANGE ON/OFF/DEFAULT
若值为ON,相当于告诉编译程序某程序某些含有复数的公式是可靠的,默认是OFF
-
for语句内的变量声明 C99中,程序员可以在for语句的初始化部分定义一个或多个变量,这些变量的作用域仅于本for语句所控制的循环体内。在C89中,这样是不可以的。 -
复合赋值初始化符 数组的格式:[index] = vol,其中index表示数组的下标,vol表示本数组元素的初始化值。 int x[10] = {[0] = 10, [5] = 30};
struct example{
int k, m, n;
} object = {m = 10, n = 200};
-
printf()和scanf()函数系列的增强 -
C99新增的标准库 -
__func__ 预定义标识符,用于指出__func__所存放的函数名
C++11
C++11是C++程序设计语言标准的一个新的版本,在2011年由ISO批准并发布。C++11新标准从而代替了原来的C++98和C++03.。C++11标准是对C++的一次巨大的改进和扩充。在核心语法,STL标准模板等方面增加众多新功能,新亮点。例如新增auto,deltype,nullptr等关键字,增加范围for循环,新增lambda表达式等。
-
auto关键字 引入了auto关键字,auto关键字可以让编译器自动分析某个初始值来判断它所属的类型。 -
decltype关键字 引入了类型说明符decltype,它使得编译器自动分析表达式的类型并得到它的类型,最关键是它不会去计算表达式的值。 -
字面值nullptr 引入了一个新的字面值来初始化空指针,nullptr是一个比较特殊的字面值,它可以任意转换成其他的任意指针类型。 -
范围for语句 范围for语句遍历指定序列的每个元素,并且可以对每个元素进行某种操作。 for(auto i: {1,2,3,4})
printf("%d\n",i);
-
Lambda表达式 C++11新标准新增的一项重要功能就是lambda表达式,表示一个可调用的代码单元,也可以理解为一个没有命名的内联函数。Capture list 表示捕获列表,也就是lambda所在函数中的局部变量的列表。Return type 表示该lambda的返回类型,Parameter list 为形参列表,Function body是函数体。另外,lambda必须包括捕获列表和函数体,另外的几个可以省略。 [capture list] (parameter list) -> return type { function body}
-
initializer_list Initializer_list定义在c++11新标准新引入的initializer_list头文件中,此类型用于访问c++初始化列表中的值,列表中的元素类型为const的。这种类型的对象由编译器从初始化列表声明中直接自动构造,所谓初始化列表声明就是被包括在花括号里面的,用逗号分隔元素的列表。 int a[3]{1,2,3};
-
智能指针shared_ptr,unique_ptr -
标准库bind函数
C99和C++11相互调用
集成开发环境(IDE,Integrated Development Environment )是用于提供程序开发环境的应用程序,一般包括代码编辑器、 编译器 、 调试器和图形用户界面等工具,是集成了代码编写功能、分析功能、编译功能、调试功能等一体化的开发软件服务套。
-
gcc :Linux系统的默认C/C++编译器,可编译.c 和.cpp 。gcc 编译器支持C99 标准和C++11 标准 -
g++ :实际调用gcc 编译.cpp 文件 -
Dev-C++ :内嵌gcc 编译器(gcc 编译器的 Windows 移植版) -
VS2017 :对于C99 ,不支持VLA变长数组,不支持指针的隐式类型转换,其他的没有测试。更重要的,它也不支持内联汇编代码,不识别__asm__ 符号。对于C++11 支持良好。
C++11调用C99
-
为C99 代码编写头文件,应符合C89 标准以兼容C++11 。包含头文件时,加上extern "C" ,告诉编译器用C的规则去调用函数。 extern "C"{
#include "xxx.h"
}
-
将C99 代码编译为动态库.dll 或者静态库.lib 。Dev 中,
- 文件 - 新建 - 项目 - C项目 - 空工程
- 项目 - 项目属性 - 普通 - 类型 - 静态库/动态库
- 项目 - 项目属性 - 编译器 - 代码生成 - 语言标准 - C99
- 全部重新编译,得到
.lib 或者.dll -
在C++11 工程的库路径里,添加上述库文件。
- 文件 - 新建 - 项目 - C++项目 - 空工程
- 项目 - 项目属性 - 普通 - 类型 - 控制台程序
- 项目 - 项目属性 - 编译器 - 代码生成 - 语言标准 - C++11
- 项目 - 项目属性 - 参数 - 链接 - 添加库或者对象
- 添加库的头文件,编写调用库函数的代码
- 全部重新编译,得到C++11的可执行程序
-
执行C++11 程序
C99调用C++11
-
为C++11 代码编写头文件,应符合C89 标准以兼容C99 。头文件中的函数声明要加上extern "C"{...} ,告诉编译器用C的规则编译函数。并且加上__cdecl ,使得编译中函数名不发生改变。 extern "C"{
void __cdecl func();
}
-
将C++11 代码编译为动态库.dll 或者静态库.lib 。Dev 中,
- 文件 - 新建 - 项目 - C++项目 - 空工程
- 项目 - 项目属性 - 普通 - 类型 - 静态库/动态库
- 项目 - 项目属性 - 编译器 - 代码生成 - 语言标准 - C++11
- 全部重新编译,得到
.lib 或者.dll -
在C99 工程的库路径里,添加上述库文件。
- 文件 - 新建 - 项目 - C项目 - 空工程
- 项目 - 项目属性 - 普通 - 类型 - 控制台程序
- 项目 - 项目属性 - 编译器 - 代码生成 - 语言标准 - C99
- 项目 - 项目属性 - 参数 - 链接 - 添加库或者对象
- 添加库的头文件,编写调用库函数的代码
- 全部重新编译,得到C99的可执行程序
-
执行C99 程序
问题
- 似乎
C++11 的静态库编译得到的xxx.a 文件无法在C99 中使用,报错new , delete 未定义(这是为什么?) C++11 的动态库编译会得到三个文件libxxx.a , libxxx.def , xxx.dll ,C99 中使用静态库时可以添加libxxx.a 库文件- 这样不会报
new , delete 未定义的错误了,但是有两个monstratup , _mcleanup 重定义的错误。不知道么的,昨天一顿操作,这两个错误消失了,成功在C99 上执行C++11 的库代码。但今天一试,又出现这两个重定义的错误了~~~ - OHHH,我知道了,
monstratup 这些函数是用来性能分析的,在.lib 生成的时候,记得把设置:项目 - 项目属性 - 编译器 - 代码性能 - 性能分析 - No。只在exe工程里,将上述的设置为Yes。
|