概念concept和requires ---C++ 20

概念conceptrequires —C++ 20

concept

concept简化了模板编程的难度

我们可以使用**concept定义模板形参的约束条件T** 模板实力替换T后必须满足std::is_integral_v<C>;true

例子:

image-20220518224709352

requires关键字可以直接约束模板形参T

如下:

image-20220518225045944

template <class C>
concept IntegerType = std::is_integral_v<C>

其中IntegerType是概念名,这里的std::is_integral_v<C>称为约束表达式,是一个纯右值的常量表达式,在编译期确定并且支持合取和析取

如果约束表达式为true,就表示任意类型都可以

如下


template <typename T>
concept Type = true;template <Type T>
struct Z
{
};int main(int argc, char* argv[])
{Z z = Z<int>();}

requires

constexpr bool bar() { return true; }template <class T>
requires (bar()) // bar()不是初等表达式,不符合语法规则,所以加个括号
struct X {};

requires子句除了能出现在模板形参列表尾部,还可以出现在函数模板声明尾部,所以下面的用法都是正确的:

template <class T> requires std::is_integral_v<T>
void foo();template <class T>
void foo() requires std::is_integral_v<T>;

那么如何确定多种约束的优先级呢?

例如:

template <class C>
concept ConstType = std::is_const_v<C>;template <class C>
concept IntegralType = std::is_integral_v<C>;template <ConstType T>requires std::is_pointer_v<T>
void foo(IntegralType auto) requires std::is_same_v<T, char* const>
{
}int main(int argc, char* argv[])
{foo<int>(1.5);
}

image-20220518233412354

编译器究竟应该用什么顺序检查约束条件呢?事实上,标准文档给出了明确的答案,编译器应该按照以下顺序检查各个约束条件。

1.模板形参列表中的形参的约束表达式,其中检查顺序就是形参出现的顺序。也就是说使用concept定义的概念约束的形参会被优先检查,放到刚刚的例子中foo<int>();最先不符合的是ConstType的约束表达式std::is_const_v<C>

2.模板形参列表之后的requires子句中的约束表达式。这意味着,如果foo的模板实参通过了前一个约束检查后将会面临std::is_pointer_v<T>的检查。

3.简写函数模板声明中每个拥有受约束auto占位符类型的形参所引入的约束表达式。还是放到例子中看,如果前两个约束条件已经满足,编译器则会检查函数实参是否满足IntegralType的约束。

4.函数模板声明尾部requires子句中的约束表达式。所以例子中最后检查的是std::is_same_v<T,char * const>

原子约束

concept格式

template < template-parameter-list >
concept  concept-name = constraint-expression;

原子约束由一个表达式E,以及E的参数映射(parameter mappings)。E的参数映射的意思是 约束的实体的跟表达式E相关的模板参数和模板类型。

原子约束是在编译器进行constraint normalization时生成的。表达式E中必然不包含 AND 或者OR的逻辑,否则它就会被分割为2个原子约束。

原子约束是表达式和表达式中模板形参到模板实参映射的组合(简称为形参映射)。比较两个原子约束是否相同的方法很特殊,除了比较代码上是否有相同的表现,还需要比较形参映射是否相同,也就是说功能上相同的原子约束可能是不同的原子约束

template <int N> constexpr bool Atomic = true;
template <int N> concept C = Atomic<N>;
template <int N> concept Add1 = C<N + 1>;
template <int N> concept AddOne = C<N + 1>;
template <int M> void f()requires Add1<2 * M> {};
template <int M> void f()requires AddOne<2 * M> && true {};int main(int argc, char* argv[])
{f<0>(); 
}

Add1AddOne其实是一样的约束,原子约束都是concept C = Atomic<N>,形参映射为N~2*M+1

image-20220519001014510

template <class T> concept sad = false;
template <class T> int f1(T) requires (!sad<T>) { return 1; };
template <class T> int f1(T) requires (!sad<T>) && true {return 2; };f1(0); // 编译失败

逻辑否定表达式是一个原子约束。所以以上代码会产生二义性,修改:

template <class T> concept not_sad = !sad<T>;
template <class T> int f2(T) requires (not_sad<T>) { return 3; };
template <class T> int f2(T) requires (not_sad<T>) && true  { return 4; };f2(0);

template int f2(T) requires (not_sad) { return 3; };
template int f2(T) requires (not_sad) && true { return 4; };

f2(0);


还有点没搞明白,稍后再看看吧

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

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

相关文章

辞职中

不知道为什么就不想再做下去了&#xff0c;说不上来工作有什么好&#xff0c;当然也说不上来什么不好。SP这个行业&#xff0c;可能不太适合我。准备走了&#xff0c;考研还是继续找工作&#xff1f;我不知道该怎么走&#xff01;看着前面都是路却找不到一条适合自己的。转载于…

向DataGridView中添加新的一行数据,可以添加到最后一行或作为第一行

我的开发环境&#xff1a;Microsoft Visual Studio .net 2005 这个程序是Windows Forms Application 新建一个Windows Forms Application项目&#xff0c;打开Form1&#xff0c;在窗体上放一个DataGridView控件和Button,在DataGridView的Columns中添加两列&#xff0c;Name分别…

Linq学习

IEnumerable<DataRow> cbCurrent from dr in cbRows.AsEnumerable() where dr.Field<string>("ObjectNO").ToString() row["运输单号"].…

非类型模板参数(参考《C++ Templates 英文版第二版》)

非类型模板参数(参考《C Templates 英文版第二版》) Chapter 3 3.1 非类型类模板参数 与前几章的简单例子不同,你也可以通过std::array实例化一个固定大小的栈,这样做的优点在于内存管理, #include <array> #include <cassert>template<typename T, std::si…

I AM NOTHING vs I AM SOMETHING

女友推荐链接&#xff1a;BBS上一个女生的创业感言 自创业的一路艰辛 http://bbs.trendsmag.com/showthread.php?t163960

DataGridView添加一行数据、全选、取消全选、清空数据、删除选中行

.net 2005下的Windows Form Application,一个DataGridView控件和4个Button&#xff0c;界面设置如下&#xff1a; 代码如下&#xff0c;有注解&#xff0c;相信大家都看得明白&#xff1a; using System;using System.Collections.Generic;using System.ComponentModel;using S…

类型萃取类型检查 Type-Traits LibraryType Checks --- C++20

类型萃取:类型检查 Type-Traits Library:Type Checks — C20 Type-Traits library 在C11的时候就已经发布,但依然随着C版本在不断更新 类型检查 Type Checks 每种类型就是十四种主要类型之一 主要类型 template <class T> struct is_void; template <class T>…

转:如何在 LoadRunner 脚本中做关联 (Correlation)

如何在 LoadRunner 脚本中做关联 (Correlation) 当录制脚本时&#xff0c;VuGen会拦截client端&#xff08;浏览器&#xff09;与server端&#xff08;网站服务器&#xff09;之间的对话&#xff0c;并且通通记录下来&#xff0c;产生脚本。在VuGen的Recording Log中&#xff0…

开博碎语

结束了5月26号的软考&#xff0c;就萌生了建一个技术博客的想法-----技术或许太空泛&#xff0c;其实就是把工作中&#xff0c;学习上技术方面的一些资料&#xff0c;一些体会汇聚一起&#xff0c;呈现出来&#xff0c;博客当然是个不错的选择。baidu一下&#xff0c;技术博客为…

DateTime时间的比较问题

关于DateTime时间的比较问题&#xff0c;我现在不清楚使用> < 和DateTime.Compare(t1, t2)、t1.CompareTo(t2)的区别。 下面是一个简单的测试程序&#xff0c;至少到现在为止&#xff0c;我还没有发现这两种比较的区别之处&#xff0c;大家有任何自己的想法&#xff0c;…

模板元编程 Template Metaprogramming--- C++ 20

模板元编程(一) Template Metaprogramming— C 20 在编译期进行类型操作 举个例子: std::move在概念上应该这样实现(实际并不是这么做的): static_cast<std::remove_reference<decltype(arg)>::type&&>(arg);意义上,std::move首先获取它的参数arg,推断…

6月,启蒙篇

6月&#xff0c;哈哈 在过去的4月与5月里我一直放荡着自己&#xff0c;我发现一个人没有目标与理想后是这样的无耐 &#xff0c;也是这样的无所做为。在那两个月里我放弃了所有的程序&#xff0c;我只是在计算机前玩游戏 。我原本以为这样会过得开心点&#xff0c;会让自己放松…

JS 计算日期天数差

function dayDiffer(startDate,endDate){console.info((endDate.getTime - startDate.getTime())/(24*60*60*1000));return Math.floor((endDate.getTime() - startDate.getTime())/(24*60*60*1000)); }转载于:https://www.cnblogs.com/jsczljh/p/3647395.html

自定义控件-实现TextBox的禁止粘贴

开发环境&#xff1a;Visual Studio .net 2005 Windows XP sp2 professional 新建->项目&#xff0d;>Windows控件库: 新建一个类&#xff0c;继承自TextBox类&#xff0c;具体源代码如下&#xff1a; using System;using System.Collections.Generic;using System.Comp…

模板元编程(二) Template Metaprogramming ---C++ 20

模板元编程(二) Template Metaprogramming —C 20 现在我们介绍参数与模板参数混合使用 先看一下例子: #include <iostream>int power(int m, int n) {int r 1;for (int k 1; k < n; k) r * m;return r; }template <int m, int n> struct Power {static in…

探索 ASP.NET Futures (Part 2 - Search Enabled)

在本系列的上一篇文章中&#xff0c;我们探索了ASP.NET Futures (May CTP)的SearchSiteMap功能&#xff0c;说明了如何将ASP.NET的SiteMap影射为符合Sitemaps协议的XML以便搜索引擎更好的抓取我们的站点。然而让搜索引擎更好的抓取我们的站点了&#xff0c;这部分的优化却仅仅对…