浅析decltype关键字

2024-03-14 21:04
文章标签 关键字 decltype 浅析

本文主要是介绍浅析decltype关键字,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

浅析decltype关键字

文章目录

  • 浅析decltype关键字
    • 前言
    • 1. 基本使用
      • 1.1 推导出表达式类型
      • 1.2 与using/typedef合用,用于定义类型。
      • 1.3 重用匿名类型
      • 1.4 泛型编程中结合auto,用于追踪函数的返回值类型
    • 2. 判别类型规则
    • 3. decltype的妙用
      • <1> 返回类型后置语法
      • <2> 推导lambda表达式的返回类型
      • <3> 结合STL容器使用
      • <4> 处理复杂类型
    • 总结

前言

​ 在现代C++编程中,decltype 是一个非常有用的关键字,它允许我们在编译时推导出表达式的类型。这个特性在模板编程和类型推导中尤其重要。今天,我们就来深入探讨 decltype 的基本用法、规则以及它如何提高我们代码的可读性和灵活性。


1. 基本使用

decltype的语法是:

decltype (expression)

这里的括号是必不可少的,decltype的作用是“查询表达式的类型”,因此,上面语句的效果是,返回 expression 表达式的类型。注意,decltype 仅仅“查询”表达式的类型,并不会对表达式进行“求值”

1.1 推导出表达式类型

int i = 4;
decltype(i) a; //推导结果为int。a的类型为int。

1.2 与using/typedef合用,用于定义类型。

using size_t = decltype(sizeof(0));//sizeof(a)的返回值为size_t类型
using ptrdiff_t = decltype((int*)0 - (int*)0);
using nullptr_t = decltype(nullptr);
vector<int >vec;
typedef decltype(vec.begin()) vectype;
for (vectype i = vec.begin; i != vec.end(); i++)
{
//...
}

这样和auto一样,也提高了代码的可读性。

1.3 重用匿名类型

在C++中,我们有时候会遇上一些匿名类型,如:

struct 
{int d ;double b;
}anon_s;

而借助decltype,我们可以重新使用这个匿名的结构体:

decltype(anon_s) as ;//定义了一个上面匿名的结构体

1.4 泛型编程中结合auto,用于追踪函数的返回值类型

这也是decltype最大的用途了。

template <typename T>
auto multiply(T x, T y)->decltype(x*y)
{return x*y;
}

2. 判别类型规则

对于decltype(e)而言,其判别结果受以下条件的影响:

  • 如果e是一个没有带括号的标记符表达式或者类成员访问表达式,那么的decltype(e)就是e所命名的实体的类型。此外,如果e是一个被重载的函数,则会导致编译错误。
  • 否则 ,假设e的类型是T,如果e是一个将亡值,那么decltype(e)为T&&
  • 否则,假设e的类型是T,如果e是一个左值,那么decltype(e)为T&
  • 否则,假设e的类型是T,则decltype(e)为T

标记符指的是除去关键字、字面量等编译器需要使用的标记之外的程序员自己定义的标记,而单个标记符对应的表达式即为标记符表达式。例如:

int arr[4]

则arr为一个标记符表达式,而arr[3]+0不是。

举例如下:

int i = 4;
int arr[5] = { 0 };
int *ptr = arr;
struct S{ double d; }s ;
void Overloaded(int);
void Overloaded(char);//重载的函数
int && RvalRef();
const bool Func(int);//规则一:推导为其类型
decltype (arr) var1; //int[] 标记符表达式decltype (ptr) var2;//int *  标记符表达式decltype(s.d) var3;//doubel 成员访问表达式//decltype(Overloaded) var4;//重载函数。编译错误。//规则二:将亡值。推导为类型的右值引用。decltype (RvalRef()) var5 = 1;//规则三:左值,推导为类型的引用。decltype ((i))var6 = i;     //int&decltype (true ? i : i) var7 = i; //int&  条件表达式返回左值。decltype (++i) var8 = i; //int&  ++i返回i的左值。decltype(arr[5]) var9 = i;//int&. []操作返回左值decltype(*ptr)var10 = i;//int& *操作返回左值decltype("hello")var11 = "hello"; //const char(&)[9]  字符串字面常量为左值,且为const左值。//规则四:以上都不是,则推导为本类型decltype(1) var12;//const intdecltype(Func(1)) var13=true;//const booldecltype(i++) var14 = i;//int i++返回右值

3. decltype的妙用

<1> 返回类型后置语法

在C++11中,我们可以使用 decltype 结合返回类型后置语法来精确定义函数的返回类型。

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

这里,decltype(t + u) 会自动推导出 tu 相加的结果类型。

<2> 推导lambda表达式的返回类型

当使用lambda表达式时,decltype 可以帮助我们推导出表达式的返回类型。

auto lambda =  -> decltype(x * y) 
{return x * y;
};

在这个例子中,decltype(x * y) 会推导出 intfloat 相乘的结果类型。

<3> 结合STL容器使用

在使用标准模板库(STL)容器时,decltype 可以用来推导迭代器类型,使代码更加简洁。

std::vector<int> vec;
auto it = vec.begin();
decltype(vec)::value_type value = *it;

这里,decltype(vec)::value_type 会推导出 vec 中元素的类型。

<4> 处理复杂类型

decltype 在处理复杂类型时非常有用,尤其是当类型难以手动书写时。

std::map<std::string, std::vector<int>> complexMap;
decltype(complexMap.begin()->second) vec; // 推导出 vector<int>

在这个例子中,decltype 能够帮助我们推导出 complexMapsecond 元素的类型,即 std::vector<int>


总结

decltype 是 C++11 引入的一个强大的关键字,它提供了一种类型安全的方式来推导变量和表达式的类型。通过上述示例和解释,我们可以看到 decltype 不仅增强了代码的可读性,还提供了更多的编程灵活性。希望这篇文章能帮助你更好地理解和使用 decltype,从而编写出更高质量的 C++ 代码。

这篇关于浅析decltype关键字的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/809719

相关文章

浅析Spring Security认证过程

类图 为了方便理解Spring Security认证流程,特意画了如下的类图,包含相关的核心认证类 概述 核心验证器 AuthenticationManager 该对象提供了认证方法的入口,接收一个Authentiaton对象作为参数; public interface AuthenticationManager {Authentication authenticate(Authenti

Oracle Start With关键字

Oracle Start With关键字 前言 旨在记录一些Oracle使用中遇到的各种各样的问题. 同时希望能帮到和我遇到同样问题的人. Start With (树查询) 问题描述: 在数据库中, 有一种比较常见得 设计模式, 层级结构 设计模式, 具体到 Oracle table中, 字段特点如下: ID, DSC, PID; 三个字段, 分别表示 当前标识的 ID(主键), DSC 当

关键字synchronized、volatile的比较

关键字volatile是线程同步的轻量级实现,所以volatile性能肯定比synchronized要好,并且volatile只能修饰于变量,而synchronized可以修饰方法,以及代码块。随着JDK新版本的发布,synchronized关键字的执行效率上得到很大提升,在开发中使用synchronized关键字的比率还是比较大的。多线程访问volatile不会发生阻塞,而synchronize

(入门篇)JavaScript 网页设计案例浅析-简单的交互式图片轮播

网页设计已经成为了每个前端开发者的必备技能,而 JavaScript 作为前端三大基础之一,更是为网页赋予了互动性和动态效果。本篇文章将通过一个简单的 JavaScript 案例,带你了解网页设计中的一些常见技巧和技术原理。今天就说一说一个常见的图片轮播效果。相信大家在各类电商网站、个人博客或者展示页面中,都看到过这种轮播图。它的核心功能是展示多张图片,并且用户可以通过点击按钮,左右切换图片。

JavaScript 根据关键字匹配数组项

要在JavaScript数组中根据关键字匹配项,可以使用filter方法结合一个测试函数。以下是一个示例代码,定义了一个函数findByKeyword,该函数接受一个数组和一个关键字,然后返回一个新数组,其中包含与关键字匹配的所有项。 function findByKeyword(array, keyword) {return array.filter(item => {// 假设要匹配的是对象

MySQL 的关键字

MySQL 中的关键字是数据库中具有特殊含义的保留字,它们用于定义数据库结构、操作数据库数据和控制数据库行为。关键字在 MySQL 查询中扮演着至关重要的角色,因为它们是 SQL 语句的核心组成部分。 1. 数据定义语言 (DDL) 关键字 数据定义语言 (DDL) 关键字用于定义、修改和删除数据库结构,如数据库、表和索引等。这些关键字通常用于创建、删除表结构以及修改表的列等操作。 1.1

C++中的mutable关键字详解

目录 1.概述 2.使用场景 3.示例 4.mutable修饰Lambda表达式 5.注意事项 1.概述         在C++中,mutable也是为了突破const的限制而设置的。被mutable修饰的变量,将永远处于可变的状态,即使在一个const函数中。         我们知道,被const关键字修饰的函数的一个重要作用就是为了能够保护类中的成员变量。即:该函数可以

[Python]生成器和yield关键字

生成器和yield关键字 1.生成器介绍: 概述: ​ 它指的是 generator, 类似于以前学过的: 列表推导式, 集合推导式, 字典推导式… 作用: ​ 降低资源消耗, 快速(批量)生成数据. 实现方式: ​ 1.推导式写法. my_generator = (i for i in range(5)) ​ 2.yield写法. def get_generator():for i

java基础总结11-面向对象7(super关键字)

在JAVA类中使用super来引用父类的成分,用this来引用当前对象,如果一个类从另外一个类继承,我们new这个子类的实例对象的时候,这个子类对象里面会有一个父类对象。怎么去引用里面的父类对象呢?使用super来引用,this指的是当前对象的引用,super是当前对象里面的父对象的引用。 1 super关键字测试 package cn.galc.test;/*** 父类* @autho

java基础总结08-面向对象4(static关键字)

原来一个类里面的成员变量,每new一个对象,这个对象就有一份自己的成员变量,因为这些成员变量都不是静态成员变量。对于static成员变量来说,这个成员变量只有一份,而且这一份是这个类所有的对象共享。 静态成员变量与非静态成员变量的区别 以下面的例子为例说明 package cn.galc.test;public class Cat {/*** 静态成员变量*/private static