最近在翻《c++函数式编程》的时候看到有一小节在说c++14新增了“菱形运算符”。我寻思c++里好像没什么运算符叫这名字啊,而且c++14新增的功能很少,我也不记得有添加这种语法特性。一瞬间我有些怀疑我的记忆了,所以为了查漏补缺,我写了这篇文章。

什么是菱形运算符

这个概念在Java里比较多见:

List<String> myList = new ArrayList<>();

这东西在Java里的学名是diamond operator,表示使用泛型类并且类型参数在左侧的表达式已给出因此在右侧可以省略。

简单的说就是让你少写几次重复的类型参数。因为看起来像个菱形所以得名菱形运算符。

然后我们偶尔会在c++里看到形状上很相似的东西:

std::sort(vec.begin(), vec.end(), std::greater<>());

<>出现在模板的特化中是我们所熟悉的,但这个std::greater<>()是什么呢?

c++没有菱形运算符

先说结论,从语言标准来说,c++里没有什么菱形运算符。

c++20里虽然新增了一个运算符operator<=>,但这个和所谓的菱形运算符没有任何关系。

那问题来了,std::greater<>()是什么以及为什么书里说是c++14新增的特性呢?难道书里瞎说的吗?但事实是这样的示例代码在c++14以及之后的标准下可以正常编译运行,而且这本书的质量尚可,虽然会在措辞上犯些小错(比如c++没有菱形运算符)但不至于花大篇幅去胡说八道。

当然,要想回答这个问题我们得先复习点基础知识。

<>在c++里的作用

先说结论,在c++里看到<>,绝大多数都是在为模板提供类型参数,当然这种东西我们不讨论:(a<1, 2>b),这里<>是在两个不同的表达式里。

那既然用来提供类型参数,那为什么可以啥都不提供呢?答案是有两类情况确实可以。

第一类是在函数模板上,类型参数可以自动推导时:

template <typename T>
void f(const T&)
{
std::cout << "f<T>\n";
}
template <>
void f(const int&)
{
std::cout << "f<int>\n";
} void f(const int&)
{
std::cout << "f\n";
} int main()
{
f(1); // f
f<>(1); // f<int>
f(1.2); // f<T>
}

非模板函数在重载决议中的优先级总是高于模板的,因此f(1)这样的表达式总是会用到最下面定义的那个非模板函数f。这时候我们可以用f<int>(1)来直接调用函数模板f,而函数模板的类型参数如果能从参数推导出来的话,可以不明确给出(也就是后面的f(1.2)那样的),而在我们现在这句表达式里,我们既要明确使用函数模板,又想让类型参数被自动推导,就得使用f<>(1)

另一种情况不分类模板还是函数模板,当模板的类型参数有默认值时,可以靠<>来使用这些默认值:

template <typename T = void>
struct Wrapper
{
using wrappered = T;
}; // Wrapper<> 等于 Wrapper<void>
static_assert(std::is_same_v<Wrapper<>::wrappered, Wrapper<void>::wrappered>);

在第二种情况下,因为没显示给出类型参数,且这里没法使用类型推导,因此编译器使用了类型参数的默认值,这里是void。

观察比较仔细的话其实会发现上面两种情况其实是一件事,<>相当于没有显示给出任何类型参数,于是对这些没有显示指定的类型参数,编译器会先尝试类型推导,如果没法推导则会检查这些类型参数是否有默认值,有就利用默认值。如果上面这两步都没法得到能正常使用的类型参数,模板会被SFINAE淘汰或者报出编译错误。

这并不是什么新语法,是从有模板开始就一直存在的规则。

现在我们可以看看std::greater<>()是什么了,首先std::greater是个类模板,然后它接受一个类型参数,这个参数在c++14之后有了默认值void,因此std::greater<>()std::greater<void>()

c++14中究竟添加了什么

既然c++14并没有添加“菱形运算符”,那究竟新增了什么呢?

在已经知道了std::greater<>()的真身后,找起来就很容易了,所以我很快找到了对应的新特性:n3421

这个特性是这样的:原先我们要用标准库提供的谓词模板,需要自己指明参数类型,这样写起来很麻烦而且对于那种嵌套的或者元素类型复杂的容器来说写明参数类型不仅费时而且费力,更要命的是对于map,一不小心是会有性能问题的:

for_each(map.begin(), map.end(), std::pred<std::pair<std::string, int64_t>>());

上述代码的问题在于正确的参数类型应该是std::pair<const std::string, int64_t>,我们漏掉了const,这会导致pair整个被复制一遍,性能是无比底下的。要彻底避免这种错误,就得利用自动类型推导。

然而前面说了,标准库提供的谓词基本全是类模板,类模板的模板参数要么依赖默认值要么得显示指定,怎么才能依赖自动推导呢。

于是这个新特性最精彩的地方来了:原先的模板的调用运算符不是模板参数也是定死的,但我们可以新加一个默认参数,然后针对这个默认参数的类型进行完全特化,在特化里提供一个泛型的operator(),这样就能利用函数模板来自动推导参数类型了,而且以前的代码不受影响。

默认参数的设置也是有讲究的,需要用一个谓词用不到的且不会影响老代码的类型,运气不错,void正好符合条件(void上几乎没法做什么操作,因此也不会被指定给这些谓词做类型参数),因此现在的greater的代码是下面这样的:

// 注意默认值是void
template <typename T = void> struct greater {
constexpr bool operator()(const T& lhs, const T& rhs) const
{
return lhs > rhs;
}
}; // 针对greater<void>的完全特化
template <> struct greater<void> {
template <class T, class U> auto operator()(T&& t, U&& u) const
-> decltype(std::forward<T>(t) > std::forward<U>(u))
{ return std::forward<T>(t) > std::forward<U>(u); }
};

当使用std::greater<T>()的时候,代码的逻辑和原来一样,当使用std::greater<void>()的时候,返回的Functor的函数调用运算符是个模板,可以自己推导参数类型和返回值类型。至于为啥greater<void>的内部构造可以和其他情况实例化的greater区别这么大,这个是c++的特性:模板的不同实例之间是可以异构的。

而且因为类型参数的默认值就是void,因此可以简写成std::greater<>()

所以c++14只是给标准库里可以代替运算符的模板们增加了默认类型参数和一个泛型的调用运算符,利用这些可以简化代码并确保类型安全。

真相是其实没啥菱形运算符,只是利用了以前就存在的模板的特性简化了标准库的使用,让人少写点字。达成的效果倒是和Java的菱形运算符差不多。

总结

显然书里有夸大成分,老话说尽信书不如无书,还得小心检验才是。

顺便我们复习了现代c++的重要原则:能依赖自动类型推导的地方,没必要自己手写。

因此应该多写这样的代码:std::sort(vec.begin(), vec.end(), std::greater<>());

不过还有最后一个问题,为啥不直接用lambda呢?那是因为能指定类型参数的泛型lambda要在c++20才出现,在这之前想要让lambda完全做到类型安全得费点功夫,而且lambda整体上也不如直接用标准库提供的std::greater<>()std::less<>()之类的简洁易懂。

C++里也有菱形运算符?的更多相关文章

  1. SQL Server里PIVOT运算符的”红颜祸水“

    在今天的文章里我想讨论下SQL Server里一个特别的T-SQL语言结构——自SQL Server 2005引入的PIVOT运算符.我经常引用这个与语言结构是SQL Server里最危险的一个——很 ...

  2. JavaScript学习笔记-循环输出菱形,并可菱形自定义大小

    var Cen = 6;//定义菱形中部为第几行(起始值为0) //for循环输出菱形 document.write("<button onclick='xh()'>点我for循 ...

  3. SQL Server里ORDER BY的歧义性

    在今天的文章里,我想谈下SQL Server里非常有争议和复杂的话题:ORDER BY子句的歧义性. 视图与ORDER BY 我们用一个非常简单的SELECT语句开始. -- A very simpl ...

  4. C++之运算符重载(2)

    上一节主要讲解了C++里运算符重载函数,在看了单目运算符(++)重载的示例后,也许有些朋友会问这样的问题.++自增运算符在C或C++中既可以放在操作数之前,也可以放在操作数之后,但是前置和后置的作用又 ...

  5. 转: Python 运算符与用法

    +加两个对象相加 3 + 5得到8.'a' + 'b'得到'ab'. (注意:6+'a'这样是错误的,但在PHP里这样是可以运行的) -减得到负数或是一个数减去另一个数 -5.2得到一个负数.50 - ...

  6. [python学习笔记] 运算符

    数学运算符 与大多语言相同的运算符就不介绍了.不同的地方会用 (!不同)标出 与java相同的运算符 , - , * , % , / 不同之处 除法 (!不同) /  与java不同,整数相除,结果为 ...

  7. 前端入门9-JavaScript语法之运算符

    声明 本系列文章内容全部梳理自以下几个来源: <JavaScript权威指南> MDN web docs Github:smyhvae/web Github:goddyZhao/Trans ...

  8. [转]C++之运算符重载(2)

    上一节主要讲解了C++里运算符重载函数,在看了单目运算符(++)重载的示例后,也许有些朋友会问这样的问题.++自增运算符在C或C++中既可以放在操作数之前,也可以放在操作数之后,但是前置和后置的作用又 ...

  9. C语言中while语句里使用scanf的技巧

    今天友人和我讨论了一段代码,是HDU的OJ上一道题目的解,代码如下 #include<stdio.h> { int a,b; while(~scanf("%d%d",& ...

  10. PHP运算符-算术运算符、三元运算符、逻辑运算符

    运算符是用来对变量.常量或数据进行计算的符号,它对一个值或一组值执行一个指定的操作.PHP的运算符包括算术运算符.字符串运算符.赋值运算符.位运算符.逻辑运算符.比较运算符.递增或递减运算符.错误控制 ...

随机推荐

  1. 【Spring注解驱动开发】面试官再问你BeanPostProcessor的执行流程,就把这篇文章甩给他!

    写在前面 在前面的文章中,我们讲述了BeanPostProcessor的postProcessBeforeInitialization()方法和postProcessAfterInitializati ...

  2. KingbaseES V8R6集群运维案例--cluster模式备份sys_backup.sh init故障

    KingbaseES V8R6集群运维案例--cluster模式备份sys_backup.sh init故障 案例说明: 通过脚本方式部署KingbaseES V8R6集群后,在'cluster'模式 ...

  3. .NET分布式Orleans - 8 - 贪吃蛇项目实战(准备阶段)

    到目前为止,Orleans7的核心概念基本已经学完,我准备使用Orleans7做一个项目实战,来总结自己的学习效果. 项目效果 通过Orleans7来完成一个贪吃蛇游戏,要求如下: 可以多人在线玩 贪 ...

  4. archlinux xfce 设置窗口背景颜色,QT背景颜色

    1.使用xfce主题 2.有QT背景不覆盖,使用配置 sudo pacman -S qt5-ct 3.在/etc/environment添加环境变量 QT_QPA_PLATFORMTHEME=qt5c ...

  5. 深入理解HashMap和TreeMap的区别

    目录 简介 HashMap和TreeMap本质区别 排序区别 Null值的区别 性能区别 共同点 深入理解HashMap和TreeMap的区别 简介 HashMap和TreeMap是Map家族中非常常 ...

  6. 深入理解 SQL UNION 运算符及其应用场景

    SQL UNION运算符 SQL UNION运算符用于组合两个或多个SELECT语句的结果集. 每个UNION中的SELECT语句必须具有相同数量的列. 列的数据类型也必须相似. 每个SELECT语句 ...

  7. Pandas统计计算

    基本的统计方法 Method Description count Number of non-NA values describe Compute set of summary statistics ...

  8. Sqlite数据库联合查询及表复制等详述

    外键:一般在两个表之间要建立关联时候,创建一个列创建 为外键(UserInfos-DeptId),它在另一个表必须是主键(DeptInfos-DeptId) 元素约束:主键约束:主要区别内容相同的行, ...

  9. Nacos 多个实例的服务调用失败

    在微服务开发阶段,开发人员会频繁启动服务. 这样Nacos上会经常出现一个服务存在多个实例,这是自己和其他同事都启动了同一个服务造成的. 此时使用OpenFeign对该服务进行远程调用,会有很大概率出 ...

  10. OS常用功能--持续更新

    import os # 拼接路径 print(os.path.join('a', 'b', 'c')) # 获取当前路径 print(os.getcwd()) # 在当前文件夹创建文件夹a # os. ...