C++11:兼容性

1  C++11保持与C99兼容

   1.1  用于跨平台的相关宏  __STDC_HOSTED__完整包含C库为1否则为0 ; __STDC__是否和c标准一致 ; __STDC__VERSION__ 所支持的c标准版本 ; __STDC_ISO_10646 编译环境是否符合某个版本的C++标准 ; 通过以上宏程序员通过#ifdef/#endif预处理实现跨平台

    1.2 __func__返回所在函数的名字,编译器会隐式的在函数定以后定义__func__的标识符,该标识符甚至可以在类中的成员函数中使用,但是不可以作为函数的默认参数因为在定义参数时该标识符还没有定义。

    1.3 _Pragma (字符串常量) ,和#pragma语义相同,不同的是_Pragma是个操作符因此可以用在一些宏中。头文件中定义了:_Pragma("once")告诉编译器该头文件只会编译一次。

    1.4 __VA_ARGS__用于替换变长参数宏定义中的省略号部分,如:#define 宏名(...)  printf(__VA_ARGS__) 用于打印宏参数

    1.5  C++11将会在拼接含宽窄字符串时,将char转换为wchar_t


2 long long 整型有两种:long long 和unsigned long long .例如:long long int ll_min=LLONG_MIN将是一个最小的long long整型数,可以在常数字面量后面加ll表示long long整型。相关的测试宏:LLONG_MIN, LLONG_MAX, ULLONG_MAX。


3 断言:断言就是将一个返回值总是需要为真的判别式放在语句中用于在设计的逻辑上不应该产生的情况。运行时断言assert宏,可以用定义#define NDEBUG禁止assert,此后assert将会是无意义的c语句。编译时断言boost库的BOOST_STATIC_ASSERT,c++11有了static_assert(常量表达式),该宏在编译时检查故称为静态断言。


4 noexcept(常量表达式) ,用于函数后表示该函数是否会抛出异常。默认noexcept等于noexecpt(true)表示不抛出异常,如函数仍会抛出异常则编译器调用std::terminate()暴力简单的终止程序,可以有效的防止异常的传播与扩散,如throw()也不允许函数抛出异常但是会有一些额外的开销如展开函数帧,并调用本帧中已构造的自动变量的析构函数等。noexcept(false)函数可能抛出异常。noexcept可以一个操作符可以用于模板,如下:

template<typename T> 
void fun() noexcept(noexcept(T())){}//第二个noexcept是个操作符,fun是否抛出异常取决于T()是否抛出异常,如T()抛出异常则第二个noexcept返回false
c++11默认将delete函数设置为noexcept,(delete跟在类的成员函数后表示类不能有该成员函数,如禁止copy constructor)。类的析构函数同样默认也是noexcept(true)的,若程序员显示指定了析构函数noexcept(false)析构函数还是会抛出异常的,且若类的基类或成员有noexcept()则该类也是noexcept()的。


5 快速初始化成员变量:在C++98中类中只有const static的整型或枚举能在类中初始化,其它staic成员在类外显示初始化,非static成员在构造函数中初始化(可以用初始化列表)。这在类中若有多个成员具有默认值且构造函数有多个的时候可能需要多次重复的书写初始化列表。C++11中可以用=和{}初始化一般成员。如下:

class A;
class test{
   public:
         test(int i):data(i){}//初始化列表可以和就地初始化共存,但是初始化列表将覆盖就地初始化的值
   private:
         int data=10;//int data{10};就地初始化
         A a{10,20};//a(10,20)无法通过编译的,只能用{}列表初始化,这里假设A构造时接收两个整数
}
这里需要注意的对于类的非const的static成员还是需要在类外定义初始化,这样才能保证静态成员最后只存在一个目标文件中。


6 sizeof操作符,在C98中sizeof不能用于非静态成员,普通成员通常含有this指针。但是在C++11中sizeof针对类的非static成员是合法的。


7 friend在C++11中友元(友元函数或类可以访问类的任意属性)的声明可以省去class关键字,这个特性可用于测试代码中,如下:

tempalte<typename T>
class A{
   public:
         friend T;
};
class test{};
A<int> one;//普通的A对象,实例化为一个内置类型表示A没有友元
A<test> two;//test是A的友元,这样测试类test就可以任意获取A的属性或方法用于测试


8 final/override控制多态,在一个基类中若不想子类重写基类中的某个成员函数则在该成员函数后面加上final,子类重写该成员函数将会编译出错,final用于终止重写。在子类的成员函数后面加上override表示这是重写基类的virtual函数,这个作用时用于编译时检查参数、返回值等是否重写时错误,用于帮助检查重写时的一些书写错误。


9 C98中模板类可以有默认模板参数,且默认模板参数通常置于后面,但是模板函数却不能有默认参数。在C++11中模板函数可以默认模板参数,且参数顺序任意。


10 外部模板,考虑下面的情况:

test.h
template<typename T>
void fun(T t){}
 
test1.cpp
#include "test.h"
void fun<int> (int);//第一份模板实例化代码
fun(10);

test2.cpp
#include "test.h"
void fun<int> (int);//第二份模板实例化代码
fun(3);


某个头文件有一个公有的模板,在多个源文件中使用相同的模板实例化代码,但是实例化代码却有多份,仿照extern的手法改为:

test.h
template<typename T>
void fun(T t){}
 
test1.cpp
#include "test.h"
tempalte void fun<int> (int);//只有这一份实例化模板代码
fun(10);

test2.cpp
#include "test.h"
extern template void fun<int> (int);
fun(3);
这个用法只适用于大工程减少编译时间。


11 C98中局部类型或匿名类型不能用于模板实参,匿名类型:struct{int i;}A;。在C++11中允许以上作为模板实例化参数。






你可能感兴趣的:(C++11兼容性)