超好用的C++11新特性(一)

2020-04-16  本文已影响0人  小腊鸡

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一样简单的代码。

上一篇下一篇

猜你喜欢

热点阅读