2011年,C++国际标准委员会公布了C++11标准,这是C++98发布后13年来第一次重大修正。C++11引入了大量非常有用的特性,使代码更直观、简洁、安全、方便,使C++更加接近java、python这一类的高级语言。
1.nullptr
nullptr的出现是为了替代NULL。
传统C++中会把NULL和0视为同一种东西,尽管有些编译器把NULL处理为((void*)0),但是在处理重载函数时就会发生混乱。
void func(int);
void func(char *);
如果你使用的编译器把NULL定义为0,那么此时func(NULL)将会去调用func(int),这可能直接导致程序运行出错。
为了解决这个问题,C++11引入了nullptr,专门用来区分NULL、0。因此,给指针赋初值的时候,请使用nullptr。
int *p=nullptr;
2.类型推导
auto
过去,使用stl时会产生大量的冗余代码,但却不能删减,不仅影响效率,还特别难阅读。你可能写过这样的代码
std::map<int,std::string>::const_iterator it=m.find(1);
但是现在,你可以这样写
auto it=m.find(1);
编译器会自动推导出正确的类型
auto a=1; // int
auto b=1.1; // double
auto c="abc"; // const char*
auto d={1,2}; // std::initializer_list<int>
值得注意的是,auto是编译期推导,即在编译时编译器推导出正确的类型,将auto自动替换成对应的类型,所以当auto与模板结合时就会产生一些问题
template<typename T,typename U>
auto add(T a,U b)
{
return a+b;
}
这样写编译器会直接报错,因为在编译期,编译器无法推导出返回值的类型,也就不能替换auto。解决办法在介绍decltype时将会提到。
注意:auto不能用于函数传参,不能用于推导数组类型。
decltype
decltype关键字是为了解决auto关键字只能对变量进行类型推导的缺陷而出现的,它的用法和sizeof很相似。
当我们需要计算表达式的类型时,decltype就显得格外重要。
auto a=1;
auto b=1.1;
decltype(a+b) c;
刚刚在介绍auto的时候,有提到auto与模板结合的一些缺陷,这里使用decltype就能解决刚刚的问题。
template<typename T,typename U>
auto add(T a,U b) -> decltype(a+b)
{
return a+b;
}
C++11引入的这种新的写法叫做返回类型后置。C++14使得普通函数具有自动推导功能,因此,这样的写法也变得合法了。
template<typename T,typename U>
auto add(T a,U b)
{
return a+b;
}
3.区间迭代
基于范围的for循环
在C++98/03的规范中,如果你使用stl的迭代器迭代,那将会是一件很麻烦的事。
for(std::vector<int>::iterator i = arr.begin(); i != arr.end(); ++i)
{
std::cout << *i << std::endl;
}
有了C++11新特性以后,你可以这样写
for(auto &i : arr) // 使用了引用
{
std::cout << i << std::endl;
}
当然,不止stl的模板类可以这样写,原本基础的数组也可以这样使用。
C++11引入的基于范围的迭代写法,使我们能够写出像python一样简单的代码。