C++ 11新特性之auto和decltype

简介: C++ 11新特性之auto和decltype

概述

在C++ 11标准中,引入了两大关键类型推导机制,即:auto关键字和decltype表达式。这两个特性不仅极大地简化了代码编写,提升了可读性,还为开发者提供了更加灵活、直观的类型声明方式。本文将详细解读auto和decltype的概念、工作原理及其在实际编程中的应用。

auto的使用

auto在C++ 03中表示临时变量的语义,由于使用极少且多余,在C++ 11中已被删除。在C++ 11之前,变量的类型必须在声明时显式指定。然而,随着模板元编程和复杂类型表达式的广泛使用,手动书写完整的类型显得既冗余又容易出错。C++ 11引入的auto关键字解决了这个问题,并具有两种用途:自动类型推断和返回值占位。

1、自动类型推断。当使用auto声明变量时,编译器会根据初始化表达式自动推断变量的类型。

#include <iostream>
#include <vector>
using namespace std;
int main()
{
    std::vector<int> vctNumber = {1, 2, 3};
    // it 的类型会被推断为 std::vector<int>::iterator
    auto it = vctNumber.begin();
    return 0;
}

2、函数返回值类型推断。auto还可以用于声明函数的返回类型,尤其在配合尾置返回类型时,可以方便地推断模板函数的返回类型。

template<typename T, typename U>
auto add(T a, U b) -> decltype(a + b)
{
    return a + b;
}

3、下面是一些使用auto的示例。

// 错误示例:没有初始化表达式,无法推断出a的类型
auto a;
// 错误示例:auto临时变量的语义在C++ 11中已不存在
auto int a = 10;
auto a = 10;
auto c = 'A';
auto s("HOPE");
vector<int> vctTemp;
auto it = vctTemp.begin();
auto ptr = [](){ cout << "Hello HOPE" << endl; };


decltype的使用

decltype是一个操作符,可用于推导括号内表达式的类型,其规则如下:

1、如果表达式e是一个变量,那么就是这个变量的类型。

2、如果表达式e是一个函数,那么就是这个函数返回值的类型。

3、如果以上都不符合,则看e是左值还是右值。若e是左值,类型为T,那么decltype(e)是T&;若是右值,则是T。

decltype的精妙之处在于以下两点。

1、类型查询。decltype是一种类型指示符,它允许程序员查询并引用表达式的类型,而不执行表达式。

#include <iostream>
using namespace std;
int main()
{
        int x = 66;
        // y的类型被推断为:int
        decltype(x++) y = 100;
        // x的值仍为:66
        cout << x << endl;
        return 0;
}


2、保持表达式的引用和cv限定符(const/volatile)。decltype能准确地保留表达式的引用类型和cv限定符,这对于编写模板代码以及处理复杂的类型关系非常有用。


#include <iostream>
using namespace std;
int main()
{
    int nNumber = 0;
    const int &nRef1 = nNumber;
    // nRef2是一个const int &类型
    decltype(nRef1) nRef2 = nNumber;
    nNumber = 66;
    // 输出:66
    cout << nRef2 << endl;
    return 0;
}


总结

auto和decltype是C++ 11中增强类型安全、简化代码的重要工具,它们通过提供类型推断功能使得程序更加简洁易读,同时也提高了开发效率。熟练掌握这两种特性,有助于编写更高质量、更具扩展性的现代C++代码。


相关文章
|
4月前
|
编译器 程序员 定位技术
C++ 20新特性之Concepts
在C++ 20之前,我们在编写泛型代码时,模板参数的约束往往通过复杂的SFINAE(Substitution Failure Is Not An Error)策略或繁琐的Traits类来实现。这不仅难以阅读,也非常容易出错,导致很多程序员在提及泛型编程时,总是心有余悸、脊背发凉。 在没有引入Concepts之前,我们只能依靠经验和技巧来解读编译器给出的错误信息,很容易陷入“类型迷路”。这就好比在没有GPS导航的年代,我们依靠复杂的地图和模糊的方向指示去一个陌生的地点,很容易迷路。而Concepts的引入,就像是给C++的模板系统安装了一个GPS导航仪
167 59
|
3月前
|
安全 编译器 C++
【C++11】新特性
`C++11`是2011年发布的`C++`重要版本,引入了约140个新特性和600个缺陷修复。其中,列表初始化(List Initialization)提供了一种更统一、更灵活和更安全的初始化方式,支持内置类型和满足特定条件的自定义类型。此外,`C++11`还引入了`auto`关键字用于自动类型推导,简化了复杂类型的声明,提高了代码的可读性和可维护性。`decltype`则用于根据表达式推导类型,增强了编译时类型检查的能力,特别适用于模板和泛型编程。
33 2
|
4月前
|
存储 编译器 C++
【C++】面向对象编程的三大特性:深入解析多态机制(三)
【C++】面向对象编程的三大特性:深入解析多态机制
|
4月前
|
存储 编译器 C++
【C++】面向对象编程的三大特性:深入解析多态机制(二)
【C++】面向对象编程的三大特性:深入解析多态机制
|
4月前
|
编译器 C++
【C++】面向对象编程的三大特性:深入解析多态机制(一)
【C++】面向对象编程的三大特性:深入解析多态机制
|
4月前
|
C++
C++ 20新特性之结构化绑定
在C++ 20出现之前,当我们需要访问一个结构体或类的多个成员时,通常使用.或->操作符。对于复杂的数据结构,这种访问方式往往会显得冗长,也难以理解。C++ 20中引入的结构化绑定允许我们直接从一个聚合类型(比如:tuple、struct、class等)中提取出多个成员,并为它们分别命名。这一特性大大简化了对复杂数据结构的访问方式,使代码更加清晰、易读。
64 0
|
4月前
|
存储 编译器 C++
【C++】面向对象编程的三大特性:深入解析继承机制(三)
【C++】面向对象编程的三大特性:深入解析继承机制
|
2天前
|
编译器 C语言 C++
类和对象的简述(c++篇)
类和对象的简述(c++篇)
|
1月前
|
C++ 芯片
【C++面向对象——类与对象】Computer类(头歌实践教学平台习题)【合集】
声明一个简单的Computer类,含有数据成员芯片(cpu)、内存(ram)、光驱(cdrom)等等,以及两个公有成员函数run、stop。只能在类的内部访问。这是一种数据隐藏的机制,用于保护类的数据不被外部随意修改。根据提示,在右侧编辑器补充代码,平台会对你编写的代码进行测试。成员可以在派生类(继承该类的子类)中访问。成员,在类的外部不能直接访问。可以在类的外部直接访问。为了完成本关任务,你需要掌握。
68 19
|
1月前
|
存储 编译器 数据安全/隐私保护
【C++面向对象——类与对象】CPU类(头歌实践教学平台习题)【合集】
声明一个CPU类,包含等级(rank)、频率(frequency)、电压(voltage)等属性,以及两个公有成员函数run、stop。根据提示,在右侧编辑器补充代码,平台会对你编写的代码进行测试。​ 相关知识 类的声明和使用。 类的声明和对象的声明。 构造函数和析构函数的执行。 一、类的声明和使用 1.类的声明基础 在C++中,类是创建对象的蓝图。类的声明定义了类的成员,包括数据成员(变量)和成员函数(方法)。一个简单的类声明示例如下: classMyClass{ public: int
50 13