C/C++语言基础--C++模板与元编程系列六,C++元编程相关库的讲解与使用

本专栏目的

  • 更新C/C++的基础语法,包括C++的一些新特性

前言

  • 模板与元编程是C++的重要特点,也是难点,本人预计将会更新10期左右进行讲解,这是第六期,讲解元编程相关库等,本人感觉这一部分内容还是比较复杂的;
  • C语言后面也会继续更新知识点,如内联汇编;
  • 欢迎收藏 + 关注,本人将会持续更新。

文章目录

    • 1、元编程简介
      • 简介
      • constexpr
      • 为什么需要元编程技术??
    • 2、C++元编程相关库的使用
      • is_void
      • is_integral
      • is_floating_point
      • is_const
      • remove_const
      • remove_reference
      • is_same
      • enable_if
      • conditional
      • declval

1、元编程简介

简介

🃏 官方:元编程(Metaprogramming)是指某类计算机程序的编写,这类计算机程序编写或者操纵其他程序(或者自身)作为它们的数据,或者在运行时完成部分本应在编译时完成的工作。

🛰 简单的说:在编译期进行计算或者类型判断这两类

constexpr

constexpr变量模板这一届已经讲过了,详细请看:C/C++语言基础–C++模板与元编程系列三(变量模板、constexpr、萃取等…………)

constexpr就是一个很典型的例子,他修饰的函数、变量这些在编译期就完成了计算或者类型判断,这里仅作简单介绍,这里举两个例子:

1、结合模板

template <int x>
struct MM {constexpr static int y = x + 1;
};
  • 这里定义了一个结构体,在结构体中定义了constexpr变量,这说明这个变量的值在编译的时候就计算出来了,但是这里要注意的是要加static修饰,在C++17标准之后,constexpr 就已经默认自动会加上static修饰,==原因是:==如果不加那么程序就不知道这个变量应该在什么地方存储,把他定义为静态变量也符合元编程的定义,🐯 提示:一般程序的内存分配是在运行的时候分配的,但是有些不是,是在编译期确定,如:静态、全局、常量等…………
  • 传递参数:是在模板中,利用模板的特化特性

2、constexpr修饰函数

constexpr int add(int a, int b)
{return a + b;
}
  • 注意constexpr修饰的的函数,那么调用他的地方,仅仅说明可以在编译期进行计算,但是不一定就在编译器进行计算,还可能在运行期计算,如下:

  • int x = 10;
    int y = 20;
    cout << add(x, y) << endl;        // 运行期计算,因为编译器不确定cout << add(10, 20) << endl;       // 编译期计算
    constexpr int sum = add(10, 20);   // 返回值结=接收,需要用constexpr修饰
    
  • 但是小编认为,最好先用变量接收这个值,然后在输出,然后需要注意的是: 返回值一定要用constexpr修饰,这样的add(x, y)会报错,这样符合元编程概念:在编译期进行静态计算.

在这里插入图片描述

为什么需要元编程技术??

主要目的: 提高效率

😈 假设:有一个东西,在运行的时候经常要用,这个值是固定的,但是需要通过计算得出,我们这里假设斐波那契数列前30项是固定的,需要经常使用,那么这个时候就可以现用元编程的循环结构先在编译器进行计算,后面在运行的时候直接调用即可(这个代码见下一节:元编程的三种编程方式)

2、C++元编程相关库的使用

C++11标准引入了一系列增强模板元编程特性的功能和库,但并没有直接引入一个名为“元编程库”的单一头文件,然而,C++11通过增强模板、类型特征(type traits)和其他编译时特性,为元编程提供了更强大的支持。

在C++11中,与元编程紧密相关的头文件主要是<type_traits>。这个头文件包含了大量的类型特征模板,这些模板可以在编译时查询类型的信息,并根据这些信息进行条件编译或生成代码

<type_traits> 头文件主要解决的是: 元编程中类型判断问题。

is_void

is_void 是在编译期进行判断,这个变量类型是否是void,一般结合模板使用(这里简单举例了一下)。

适用场景

  • 检查基本类型
  • 检查自定义类型
#include <iostream>
#include <iomanip>
#include <type_traits>using namespace std;struct MyType {string name;unsigned int age;
};int main()
{// 检查基本类型cout << boolalpha;cout << "is_void<void>: " << is_void<void>::value << endl;cout << "is_void<int>: " << is_void<int>::value << endl;// 检查自定义类型cout << "is_void<MyType>: " << is_void<MyType>::value << endl;// C++17简化版本cout << "C++17: " << is_void_v<void> << endl; return 0;
}

输出:

在这里插入图片描述

is_integral

is_void 是在编译期进行判断类型是否为整型类型

注意:整型类型包括所有的整数类型(如 charshortintlonglong long 等)以及它们的无符号版本(如 unsigned charunsigned shortunsigned intunsigned longunsigned long long 等),不只是int哈 🤠🤠🤠🤠

#include <iomanip>
#include <iostream>
#include <type_traits>using namespace std;int main()
{// 判断是整型cout << boolalpha;cout << "is_integral<int>: " << is_integral<int>::value << endl;cout << "is_integral<char>: " << is_integral<char>::value << endl;cout << "is_integral<unsigned>: " << is_integral<unsigned>::value << endl;cout << "is_integral<long long>: " << is_integral<long long>::value << endl;// 判断不是整形cout << "is_integral<string>: " << is_integral<string>::value << endl;cout << "is_integral<double>: " << is_integral<double>::value << endl;// C++17 优化版本cout << "C++17: " << is_integral_v<int> << endl;return 0;
}

输出:

在这里插入图片描述

is_floating_point

用于在编译时检查一个给定的类型是否为浮点型类型。浮点型类型主要包括 floatdouble .

#include <iostream>
#include <iomanip>
#include <type_traits>using namespace std;int main()
{// 检查浮点型cout << boolalpha;cout << "is_floating_point<float>: " << is_floating_point<float>::value << endl;cout << "is_floating_point<double>: " << is_floating_point<double>::value << endl;// 检查非浮点型cout << "is_floating_point<string>: " << is_floating_point<string>::value << endl;cout << "is_floating_point<int>: " << is_floating_point<int>::value << endl;// C++17优化版本cout << "C++17: " << is_floating_point_v<int> << endl;return 0;
}

输出:

在这里插入图片描述

is_const

编译时检查一个给定的类型是否被const修饰。如果类型是常量类型,则返回为 true;否则为 false

#include <iostream>
#include <iomanip>
#include <type_traits>using namespace std;template<typename _Ty>
void isConst()
{// 17版本: is_const_vif (is_const<_Ty>::value) {cout << "is const" << endl;}else {cout << "no const" << endl;}
}void test()
{isConst<int>();isConst<const int>();isConst<const char*>();
}int main()
{test();return 0;
}

输出:

在这里插入图片描述

remove_const

在编译时移除类型的 const 限定符

  • 如果给定的类型已经是 const 限定的,std::remove_const 会生成一个去除了 const 限定符的新类型
  • 如果给定的类型原本就没有 const 限定符,那么它就直接返回该类型本身
#include <iostream>
#include <iomanip>
#include <type_traits>using namespace std;template<typename _Ty>
void fix(_Ty t)
{// C++14 简化版本: remove_const_tusing Type = remove_const<_Ty>::type;Type num = t;cout << num + 10 << endl;
}int main()
{constexpr int x = 10;fix<decltype(x)>(x);return 0;
}

结果:

在这里插入图片描述

remove_reference

在编译时移除类型的引用限定符(即 &&&

  • 如果给定的类型是一个引用类型,std::remove_reference 会生成一个去除了引用限定符的新类型
  • 如果给定的类型原本就不是引用类型,那么它就直接返回该类型本身。
#include <iostream>
#include <iomanip>
#include <type_traits>using namespace std;template <class _Ty>
void fix(_Ty t) 
{if (is_same_v<_Ty, int&>) {t += 10;cout << "before t: " << t << endl;// C++简化版: remove_reference_tusing Type = remove_reference<_Ty>::type;Type num = t;num += 20;cout << "before t: " << t << "\tbefore num: " << num << endl;}
}int main()
{int x = 20;int& y = x;fix<int&>(y);return 0;
}

结果:

在这里插入图片描述

👀 很明显,这个去掉了&d的作用

is_same

用于在编译时检查两个类型是否完全相同

不是直接可执行的代码,而是用作模板参数或与其他模板结合使用时,在编译时提供类型信息的工具

#include <iostream>
#include <iomanip>
#include <type_traits>using namespace std;template <typename _Ty>
void fix(_Ty t)
{// 如果不用constexpr会报错,因为不用constexpr编译器会在运行的时候运行代码,这个时候编译器就是错误if constexpr (is_same_v<_Ty, string>) {t += "--fix";}else {t += 10;}cout << " fix: " << t << endl;
}void test()
{fix<int>(18);fix<string>(string("yxz"));fix<double>(8);
}int main()
{cout << boolalpha;test();return 0;
}

结果:

在这里插入图片描述

😜 这里有个注意点if constexpr结构,这个下一篇介绍元编程的三种编程方式会讲🏚🏚

enable_if

用于在编译时基于模板参数的条件来选择性地启用或禁用函数模板、类模板的特定特化—,或者模板成员函数。std::enable_if 通常与 typename = std::enable_if<condition>::type(C++11),在C++14有enable_if_t

💌 核心: 启用、禁止

#include <iostream>
#include <type_traits>using namespace std;template<typename _Ty, typename = enable_if<is_integral_v<_Ty>>::type>
void printInt(_Ty t)
{cout << "整型数据: " << t << endl;
}int main()
{printInt<int>(10);//printInt<double>(10.0);  // 报错return 0;
}

结果:

在这里插入图片描述

conditional

std::conditional 是一个模板结构体,它根据给定的条件来选择两个类型之一。这类似于条件编译指令(如 #ifdef),但它是在类型级别上工作的,而不是在编译指令级别。

std::conditional 模板接受两个类型参数和一个布尔条件作为模板参数

  • 如果条件为真(true),则 std::conditionaltype 成员别名将解析为第一个类型参数;
  • 如果条件为假(false),则它将解析为第二个类型参数。
#include <iostream>
#include <type_traits>using namespace std;int main()
{constexpr bool condition = false;   // 条件判断, 这个一般需要结合模板, 然后更具实际业务常见using Type = conditional<condition, int, string>::type;Type t;if constexpr (!is_integral_v<Type>) {t += string("yxz");}cout << t << endl;return 0;
}

结果:

在这里插入图片描述

declval

std::declval 是 C++11 引入的一个非常有用的工具,它位于 <utility> 头文件中,它允许在编译期的时候判断一个类是不是包含某一种方法

std::declval 基本用法

template<typename T>  
auto func() -> decltype(std::declval<T>().member_function());   //判断T这个类是否有member_function这个方法

🛰 例子:

#include <iostream>
#include <type_traits>
#include <utility>using namespace std;class Test
{
public:void print(){cout << __FUNCTION__ << endl;}
};class Other
{
public:Other(){cout << __FUNCTION__ << endl;}
};template <typename T>
auto func() -> decltype(declval<T>().print())
{cout << __FUNCTION__ << endl;
}int main()
{func<Test>();//func<Other>();  报错return 0;
}

结果:

在这里插入图片描述

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.rhkb.cn/news/471517.html

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

uni-app之数据驱动的picker选择器( uni-data-picker)之可以选择到任意级别

背景说明 uni-app 官方的插件市场有数据驱动选择器&#xff0c;可以用作多级分类的场景。本人引入插件后&#xff0c;发现&#xff0c;在h5和微信小程序都只能选择到叶子级。而在给出的官方组件示例中确并非如此。 以选择年级&#xff0c;而不选择班级。然后&#xff0c;想试试…

探索 HTML 和 CSS 实现的蜡烛火焰

效果演示 这段代码是一个模拟蜡烛火焰的HTML和CSS代码。它创建了一个具有动态效果的蜡烛火焰动画&#xff0c;包括火焰的摆动、伸缩和光晕的闪烁。 HTML <div class"holder"><div class"candle"><div class"blinking-glow"&g…

react + ts定义接口类型写法

接口&#xff08;未进行ts定义&#xff09; export async function UserList(params: {// keyword?: string;current?: number;pageSize?: number;},// options?: { [key: string]: any }, ) {return request<API1.UserList>(http://geek.itheima.net/v1_0/mp/artic…

【教程】Ubuntu设置alacritty为默认终端

转载请注明出处&#xff1a;小锋学长生活大爆炸[xfxuezhagn.cn] 如果本文帮助到了你&#xff0c;欢迎[点赞、收藏、关注]哦~ 目录 背景介绍 设置教程 注意事项 背景介绍 alacritty是一个开源的终端&#xff0c;比默认的xterm更好看&#xff0c;甚至编辑文本时候还会代码高亮…

使用Element UI实现前端分页,及el-table表格跨页选择数据,切换分页保留分页数据,限制多选数量

文章目录 一、前端分页1、模板部分 (\<template>)2、数据部分 (data)3、计算属性 (computed)4、方法 (methods) 二、跨页选择1、模板部分 (\<template>)2、数据部分 (data)3、方法 (methods) 三、限制数量1、模板部分 (\<template>)2、数据部分 (data)3、方法…

写给初学者的React Native 全栈开发实战班

React Native 全栈开发实战班 亲爱的同学们&#xff1a; 很高兴在这里与大家相聚&#xff01;我是你们的讲师&#xff0c;将带领大家一起踏上 React Native 移动开发的学习之旅。 为什么选择 React Native&#xff1f; 在这个移动互联网时代&#xff0c;App 开发工程师已经…

StarRocks Summit Asia 2024 全部议程公布!

随着企业数字化转型深入&#xff0c;云原生架构正成为湖仓部署的新标准。弹性扩展、资源隔离、成本优化&#xff0c;帮助企业在云上获得了更高的灵活性和效率。与此同时&#xff0c;云原生架构也为湖仓与 AI 的深度融合奠定了基础。 在过去一年&#xff0c;湖仓技术与 AI 的结…

[CKS] K8S Dockerfile和yaml文件安全检测

最近准备花一周的时间准备CKS考试&#xff0c;在准备考试中发现有一个题目关于Dockerfile和yaml文件安全检测的题目。 ​ 专栏其他文章: [CKS] Create/Read/Mount a Secret in K8S-CSDN博客[CKS] Audit Log Policy-CSDN博客 -[CKS] 利用falco进行容器日志捕捉和安全监控-CSDN博…

鸿蒙之多选框(Checkbox)

前言&#xff1a; 控制单个或者多个选项的选中状态&#xff0c;就可以使用 多选框组件 Checkbox:多选框组件CheckboxGroup:多选框组&#xff0c;控制多个多选框 Checkbox: 参数CheckboxOptions说明 名称 类型 必填 描述 name string 否 用于指定多选框名称。一般结合Ch…

CSP/信奥赛C++语法基础刷题训练(8):洛谷P5718:找最小值

CSP/信奥赛C语法基础刷题训练&#xff08;8&#xff09;&#xff1a;洛谷P5718&#xff1a;找最小值 题目描述 给出 n n n 和 n n n 个整数 a i a_i ai​&#xff0c;求这 n n n 个整数中最小值是什么。 输入格式 第一行输入一个正整数 n n n&#xff0c;表示数字个数。…

【云原生系列--Longhorn的部署】

Longhorn部署手册 1.部署longhorn longhorn架构图&#xff1a; 1.1部署环境要求 kubernetes版本要大于v1.21 每个节点都必须装open-iscsi &#xff0c;Longhorn依赖于 iscsiadm主机为 Kubernetes 提供持久卷。 apt-get install -y open-iscsiRWX 支持要求每个节点都安装 N…

【C++】string类(附题)

一、为什么学习string类&#xff1f; 1.1 C语言中的字符串 C语言中&#xff0c;字符串是以\0结尾的一些字符的集合&#xff0c;为了操作方便&#xff0c;C标准库中提供了一些str系列 的库函数&#xff0c;但是这些库函数与字符串是分离开的&#xff0c;不太符合OOP的思想&…

前端vue 列表中回显并下拉选择修改标签

1&#xff0c;vue数据列表中进行回显状态并可以在下拉框中选择修改&#xff0c;效果如下 2&#xff0c;vue 页面关键代码 <el-table-column label"审核" align"center" class-name"small-padding fixed-width" prop"status" >&…

Brave127编译指南 Windows篇:部署Node.js(五)

1. 概述 在Brave浏览器的编译过程中&#xff0c;Node.js扮演着关键角色。作为一个建立在Chrome V8引擎之上的JavaScript运行时环境&#xff0c;Node.js为开发者提供了在服务器端执行JavaScript代码的能力。它的非阻塞、事件驱动架构使其特别适合构建高性能、可扩展的网络应用。…

嵌入式硬件实战提升篇(一)-泰山派RK3566制作多功能小手机

引言&#xff1a;主要针对于嵌入式全栈内容的知识点汇总并对于linux等相关驱动知识点进行串联&#xff0c;用大家参考学习&#xff0c;并用到了嘉立创提供的泰山派RK3566作为学习的主控。 实物演示如下所示&#xff1a; 目录 一、硬件设计 1.转接电路 2.背光电路 3.音频接…

MySQL:数据库的约束

约束类型 NOT NULL - 指示某列不能存储 NULL 值。 UNIQUE - 保证某列的每行必须有唯一的值。 DEFAULT - 规定没有给列赋值时的默认值。 PRIMARY KEY - NOT NULL 和 UNIQUE 的结合。确保某列&#xff08;或两个列多个列的结合&#xff09;有唯一标识&#xff0c;有助于更容易更…

Ps:OpenColorIO 设置

Ps菜单&#xff1a;编辑/OpenColorIO 设置 Edit/OpenColorIO Settings 在专业的图像编辑和色彩管理工作流程中&#xff0c;准确的色彩呈现和转换至关重要。OpenColorIO&#xff08;OCIO&#xff09; 是一种开源的色彩管理框架&#xff0c;广泛应用于影视、动画和视觉特效行业。…

datastage在升级版本到11.7之后,部分在11.3上正常执行的SP报错SQLSTATE = 22007: 本机错误代码 = -180

在升级版本到11.7之后&#xff0c;部分在11.3上正常执行的SP开始报错&#xff0c;报的SQL错误是时间参数问题&#xff0c;但是一样的SP可以直接call sp执行&#xff0c;也可以手动调用作业执行&#xff0c;只有设置定时调度时作业会报错&#xff0c; CALLXXX.XXX(1,CURRENT TIM…

网络基础Linux

目录 计算机网络背景 网络发展 认识 "协议" 网络协议初识 OSI七层模型 TCP/IP五层(或四层)模型 网络传输基本流程 网络传输流程图 ​编辑 数据包封装和分用 网络中的地址管理 认识IP地址 认识MAC地址 笔记&#xff08;画的图&#xff09; 协议&#x…

【C#设计模式(4)——构建者模式(Builder Pattern)】

前言 C#设计模式(4)——构建者模式(Builder Pattern) 运行结果 代码 public class Computer {private string part1 "CPU";private string part2 "主板";private string part3 "内存";private string part4 "显卡";private st…