转眼间,C++20的标准已经发布快两年了。不少C++的开源项目也已经将标准升级到最新的C++20了,笔者也开启了新标准的学习历程了。所以借这系列的博文,记录下笔者学习新标准的一些心得与吐槽~~

作为C++20系列的第一篇开篇之文,就要从千呼万唤始处理的concept聊起了,后续很多新的feature的实现,也仰赖新的concept的实现,后续笔者的文章也会逐步展开。OK,开始我们C++20旅程的第一站:concept

1.First Look

先从一个群友的一个实际的问题出发,我们来看看concept可以解决什么问题。是怎么样通过coding实现的。

  • SFINAE

    熟悉C++模板编程的小伙伴肯定第一时间想到通过SFINAE的方式来解决,让笔者来解决这个问题的话,会写出下面的代码:
template <typename T>
T test(T a) {
static_assert(!std::is_same_v<void, decltype(a + a)>);
static_assert(!std::is_same_v<void, decltype(a - a)>);
static_assert(!std::is_same_v<void, decltype(a * a)>);
static_assert(!std::is_same_v<void, decltype(a / a)>);
return a;
}

这里写的代码是一个略微Trick的表达,利用decltype去获取操作符计算后的类型,然后用std::is_same_v进行一个其实没什么意义的类型比较,来满足static_assert的语义,最终满足我们对模板类型T的一些约束。

  • concept

    显然上面的方式是很不直观的,虽然能达到咱们的目的,但是从代码优雅角度来说是一种较差的选择实现。

我们来看一下用C++20提供给我们的Concept是如何解决这个问题的:

template <typename T>
concept Cal = requires (T a) {
a + a;
a - a;
a * a;
a / a;
}; template <typename T>
requires Cal<T>
T test(T a) {
return a;
}

这是通过concept来实现的一个类型约束方式,Cal代表着一个concept的实现,requires中花括号的内容就代表了对于类型T的约束,要满足下面的操作符

a + a;
a - a;
a * a;
a / a;

Bingo! 似乎C++20给了我们一个更好的trait,接着往下看,我们继续来细探Concept的实现。

2. How to use

  • concept的定义

这里写了一个例子,咱们基于这个例子来展开:

template <typename T>
concept Cal = requires (T a) {
a + a;
typename T::type;
{a + a} -> std::same_as<float>;
require Concept2<T>;
};

这里定义了4个requires的要求,只有满足这4个条件才能通过concept的限制,正常进行编译。

1). a + a这个是最简单的,该表达式能通过编译则代表符合要求,这里不会进行实际的计算。

2). typename T::type代表需要,T类型定义了type类型,才符合要求

3). {a + a} -> std::same_as<float> 这里的{}代表了decltype(a + a)之后的类型需要满足后面的concept的需求。这是一个语法糖,也可以通过这样来实现:requires std::is_floating_point_v<decltype(a + a)>

4). 最后的require Concept2<T>这代表了concept的嵌套逻辑。requires后面可以带任意的concept

  • concept的使用

了解了concept定义之后,我们就可以利用concept来进行模板类型的约束了。

这里“回字有四种写法”,大家可以选择自己喜欢的方式来使用。(真搞不懂搞这么多写法干什么,不能统一一下吗?, 下面介绍的顺序随笔者的偏好以此递减)

template<typename T>
requires Cal<T>
T test(T a)
{
return a + a;
}

1). 这是笔者最认可的一种书写方式,语义明确,在模板类型定义之后明确对它的要求。

template<Cal T>
T test(T a)
{
return a + a;
}

2). 也还行吧,模板参数前指定了concept,也比较明确直观。

Cal test(Cal a)
{
return a + a;
}

3). concept命名一长就显得有些琐碎了,并且把concept当类型使用了,看起来有些怪异了。

template<typename T>
T test(T a) requires Cal<T>
{
return a + a;
}

4). 把concept写后面的都是异端,当然如果你喜欢的话,也ok。

3. concept的本质

concept本质上是:一个可以套用在模板上的constexpr bool值。

相信下面这段代码能帮助你更好的理解它:

template <typename T>
concept Con = std::is_floating_point_v<T>; int main(int argc, char* argv[]) {
static_assert(std::is_same_v<decltype(Con<float>), bool>);
std::cout << Con<int> << std::endl;
return 0;
}

显然,上面的代码我们用is_same_v确定了concept生成的类型就是bool类型。而同样的,在运行期,咱们也可以将concept的结果作为一个bool常量进行使用,并打印。

所以,take it easyconcept很简单,它只是C++20给你提供的一个better的工具,来摆脱被疯狂的模板报错所支配的恐惧。但即使你完全不了解它,使用老的方式,依然能够同样解决问题。

4.小结

C++的一些模板推断的错误常常让人抓狂。而很多时候我们使用它需要

  • 要进行模板推断类型的编程设计
  • 利用SFINAE的方式来类型约束

这无形之中增加Coding时的心智成本,而concept作为一个新的语法糖,给了我们拆分二者的机会:让上帝归上帝,凯撒归凯撒

使用好concept来进行类型约束,enjoy新标准带来的便利吧。

希望大家能够有所收获,笔者水平有限。成文之处难免有理解谬误之处,欢迎大家多多讨论,指教。

5.参考资料

CppReference

C++雾中风景18:C++20, 从concept开始的更多相关文章

  1. [网站公告]18:07-18:20阿里云SLB故障造成网站不能正常访问

    (注:由于阿里云SLB管理控制台监控数据不准,实际故障时间是18:07-18:20.) 17:55-18:2018:07-18:20,我们使用的阿里云SLB(负载均衡)中有3台出现突发故障,造成全站无 ...

  2. 小明很喜欢数学,有一天他在做数学作业时,要求计算出9~16的和,他马上就写出了正确答案是100。但是他并不满足于此,他在想究竟有多少种连续的正数序列的和为100(至少包括两个数)。没多久,他就得到另一组连续正数和为100的序列:18,19,20,21,22。现在把问题交给你,你能不能也很快的找出所有和为S的连续正数序列? Good Luck!

    // test20.cpp : 定义控制台应用程序的入口点. // #include "stdafx.h" #include<iostream> #include< ...

  3. [18/11/20]break与continue的区别

    一.普通break 和continue 1.break: break用于强行退出循环,不执行循环中剩余的语句. 2.continue continue 语句用在循环语句体中,用于终止某次循环过程,即跳 ...

  4. Windows phone应用开发[18]-下拉刷新

    在windows phone 中采用数据列表时为了保证用户体验常遇到加载数据的问题.这个问题普遍到只要你用到数据列表就要早晚面对这个问题. 很多人会说这个问题已经有解决方案. 其实真正问题并不在于如何 ...

  5. 18位身份证验证--java实现,正则表达式

    简单的正则表达式: (1)preg_match("/^(\d{18,18}|\d{15,15}|\d{17,17}x)$/",$id_card)(2)preg_match(&quo ...

  6. Visual Studio原生开发的20条调试技巧(下)

    我的上篇文章<Vistual Studio原生开发的10个调试技巧>引发了很多人的兴趣,所以我决定跟大家分享更多的调试技巧.接下来你又能看到一些对于原生应用程序的很有帮助的调试技巧(接着上 ...

  7. [CareerCup] 18.4 Count Number of Two 统计数字2的个数

    18.4 Write a method to count the number of 2s between 0 and n. 这道题给了我们一个整数n,让我们求[0,n]区间内所有2出现的个数,比如如 ...

  8. 18.用两个栈实现队列[2StacksToImplementQueue]

    [题目] 某队列的声明如下:  C++ Code  123456789101112131415   template<typename T> class CQueue { public: ...

  9. 对Linux新手非常有用的 20个命令

    你打算从Windows换到Linux上来,还是你刚好换到Linux上来?哎哟!!!我说什么呢,是什么原因你就出现我的世界里了.从我以往的经验来说,当我刚使用Linux,命令,终端啊什么的,吓了我一跳. ...

随机推荐

  1. SharePoint Online 触发 Outlook 邮件内审批

    前言 我们在做SharePoint Online项目时, 经常会有客户问,我们能否在通知邮件中快速完成审批,而不是需要在邮件中打开系统,然后在系统中审批? 答案肯定是可以的,来!安排! 正文 1.我们 ...

  2. Codeforces Round #739 (Div. 3)

    A. Dislike of Threes 简单的水题,预处理即可 AC_CODE #include <bits/stdc++.h> using namespace std; templat ...

  3. 新一代Python包管理工具来了

    1 简介 说起Python的包管理工具,大家第一时间想到的肯定是pip.conda等经典工具.但最近我发现了一款新颖的Python包管理工具--pdm,它受到PEP582(https://www.py ...

  4. 集合remove()方法相关问题

    学习集合的过程中,了解到一个有关于remove()方法的有关特性,特此记录 首先remove方法的格式: collection.remove(Object o); 这是指对集合collection内的 ...

  5. ARC-124 部分题解

    E 直接统计原式不好做,注意到首先我们应该知道怎样的 \(x\) 序列是合法的,那么不妨首先来统计一下合法的 \(x\) 序列数量. 令 \(b_i\) 为 \(i\) 向右给的球数,那么有(\(i ...

  6. unicode家族

    定义 如果把各种文字编码形容为各地的方言,那么Unicode就是世界各国合作开发的一种语言. Unicode 只是一个符号集,它只规定了符号的二进制代码,却没有规定这个二进制代码应该如何存储. UTF ...

  7. Entity Framework Core的坑,Select后再对导航属性进行查询或Select前进行Skip/Take

    把asp.net core的项目发布到ubuntu上了,运行的时候出现了如下警告: warn: Microsoft.EntityFrameworkCore.Query[20500] The LINQ ...

  8. UITextView模拟UITextField 设置Placeholder属性 --董鑫

    由于最近有用到输入框,刚开始考虑的是UITextField,因为它在没有输入的时候可以有提示的Placeholder更能,很人性化,但UITextField只能单行输入,不能跳行,对于一些强迫症的亲来 ...

  9. K8s二进制部署单节点 etcd集群,flannel网络配置 ——锥刺股

    K8s 二进制部署单节点 master    --锥刺股 k8s集群搭建: etcd集群 flannel网络插件 搭建master组件 搭建node组件 1.部署etcd集群 2.Flannel 网络 ...

  10. 解决"Uncaught (in promise) Error: Navigation cancelled from "/" to "/login" with a new navigation"报错处理

    Uncaught (in promise) Error: Navigation cancelled from "/" to "/login" with a ne ...