模板参数不一定非得是类型,它们还可以是普通的数值。我们仍然使用前面文章的Stack的例子。

类模板的非类型模板参数

声明:

  1. template <typename T, std::size_t Maxsize>
  2. class Stack {
  3. private:
  4. std::array<T, Maxsize> elems; // elements
  5. std::size_t numElems; // current number of elements
  6. public:
  7. Stack(); // constructor
  8. void push(T const &elem); // push element
  9. void pop(); // pop element
  10. T const &top() const; // return top element
  11. bool empty() const { // return whether the stack is empty
  12. return numElems == 0;
  13. }
  14. std::size_t size() const { // return current number of elements
  15. return numElems;
  16. }
  17. };

实现:

  1. template <typename T, std::size_t Maxsize>
  2. Stack<T, Maxsize>::Stack() : numElems(0) {
  3. // nothing else to do
  4. }
  5. template <typename T, std::size_t Maxsize>
  6. void Stack<T, Maxsize>::push(T const &elem) {
  7. assert(numElems < Maxsize);
  8. elems[numElems] = elem;
  9. ++numElems;
  10. }

使用:

  1. Stack<int,20> int20Stack; // stack of up to 20 ints
  2. Stack<int,40> int40Stack; // stack of up to 40 ints
  3. Stack<std::string,40> stringStack; // stack of up to 40 strings

这个很简单,就不细说了。

函数模板的非类型模板参数

  1. template<int Val, typename T>
  2. T addValue (T x) {
  3. return x + Val;
  4. }
  5. std::transform(source.begin(), source.end(), dest.begin(), addValue<5, int>);

再比如标准库type_traits中的enable_ifconditional:

  1. template <bool _Bp, class _If, class _Then>
  2. struct _LIBCPP_TEMPLATE_VIS conditional {typedef _If type;};
  3. template <bool, class _Tp = void> struct _LIBCPP_TEMPLATE_VIS enable_if {};

限制

必须是下面的类型:

  • 整型常量/枚举
  • 指向对象/函数/成员变量的指针
  • 对象/函数的左值引用
  • std::nullptr_t

当传递指针或者引用时,指向的对象不能是字符串常量、临时变量、数据成 员以及其它子对象。

使用auto推断非类型模板参数

从c++17起,可以使用auto推断非类型模板参数:

  1. template <auto value> void f() { }
  2. f<10>(); // deduces int

如果没有auto,想将非类型模板参数的类型也当做模板参数,那么必须声明两个模板参数:

  1. template <typename Type, Type value> constexpr Type TConstant = value;
  2. constexpr auto const MySuperConst = TConstant<int, 100>;

从c++17开始,只需要一个auto即可:

  1. template <auto value> constexpr auto TConstant = value;
  2. constexpr auto const MySuperConst = TConstant <100>;

在auto推导的的情况下获取类型:

  1. template<auto Val, typename T = decltype(Val)>
  2. T foo();

或者:

  1. template<auto Val> struct Value {
  2. using ArgType = decltype(Val);
  3. };

template<decltype(auto) N>也是可以的,这里N会被推断成引用类型:

  1. template<decltype(auto) N>
  2. class C {
  3. ...
  4. };
  5. int i;
  6. C<(i)> x; // N is int&

(完)

朋友们可以关注下我的公众号,获得最及时的更新:

c++11-17 模板核心知识(三)—— 非类型模板参数 Nontype Template Parameters的更多相关文章

  1. c++11-17 模板核心知识(十一)—— 编写泛型库需要的基本技术

    Callables 函数对象 Function Objects 处理成员函数及额外的参数 std::invoke<>() 统一包装 泛型库的其他基本技术 Type Traits std:: ...

  2. c++11-17 模板核心知识(十二)—— 模板的模板参数 Template Template Parameters

    概念 举例 模板的模板参数的参数匹配 Template Template Argument Matching 解决办法一 解决办法二 概念 一个模板的参数是模板类型. 举例 在c++11-17 模板核 ...

  3. c++11-17 模板核心知识(十四)—— 解析模板之依赖型模板名称(.template/->template/::template)

    tokenization与parsing 解析模板之类型的依赖名称 Dependent Names of Templates Example One Example Two Example Three ...

  4. c++11-17 模板核心知识(十五)—— 解析模板之依赖型类型名称与typename Dependent Names of Types

    模板名称的问题及解决 typename规则 C++20 typename 上篇文章c++11-17 模板核心知识(十四)-- 解析模板之依赖型模板名称 Dependent Names of Templ ...

  5. C++非类型模板参数

    对于函数模板与类模板,模板参数并不局限于类型,普通值也可以作为模板参数.在基于类型参数的模板中,你定义了一些具体的细节来加以确定代码,直到代码被调用时这些细节才被真正的确定.但是在这里,我们面对的是这 ...

  6. C++ Template之非类型模板参数

    非类型模板参数是通过基本变量类型引入,例如int,在使用时必须显式自定值,不能通过推断. 非类型模板参数的限制:不能是浮点数(在vc6.0上测试可以为浮点型),对象以及指向内部链接对象的指针. #in ...

  7. C++标准库函数 end 的实现原理(非类型模板参数)

    在刚开始学习<C++ Primer>的时候遇到了 end 函数,感觉很神奇,但又很迷惑:为什么能获得数组的尾后指针呢?编译器也不会在内存中申请一块空间放数组元素的个数啊!最近再一次遇到了 ...

  8. C++ Templates(1.3 多模板参数 Multiple Template Parameters)

    返回完整目录 目录 1.3 多模板参数 Multiple Template Parameters 1.3.1 为返回类型设置模板参数参数 Template Parameters for Return ...

  9. c++11-17 模板核心知识(一)—— 函数模板

    1.1 定义函数模板 1.2 使用函数模板 1.3 两阶段翻译 Two-Phase Translation 1.3.1 模板的编译和链接问题 1.4 多模板参数 1.4.1 引入额外模板参数作为返回值 ...

随机推荐

  1. c++中 #define和const的区别

    来源参考:https://blog.csdn.net/yi_ming_he/article/details/70405364 这个区别用从几个角度来说: 角度1: 就定义常量说的话:  const 定 ...

  2. 【暂咕咕咕】SuffixTree

    #include<bits/stdc++.h> using namespace std; const int MAXN=1e6+10; typedef long long ll; char ...

  3. C# 读取路径的各种方式

    //1.获取模块的完整路径. string path1 = System.Diagnostics.Process.GetCurrentProcess().MainModule.FileName; // ...

  4. C语言编程入门之--第六章C语言控制语句

    导读:本章带读者理解什么是控制语句,然后逐个讲解C语言常用的控制语句,含有控制语句的代码量多起来后就要注意写代码的风格了,本章末节都是练习题,大量的练习才能掌握好控制语句的使用. 6.1 什么是控制语 ...

  5. 【3】Java面试-Servlet

    Servlet面试问题 Q1.什么是servlet? Java Servlet是服务器端技术,通过提供对动态响应和数据持久性的支持来扩展Web服务器的功能. javax.servlet和javax.s ...

  6. RHSA-2017:2029-中危: openssh 安全和BUG修复更新(存在EXP、代码执行、本地提权)

    [root@localhost ~]# cat /etc/redhat-release CentOS Linux release 7.2.1511 (Core) 修复命令: 使用root账号登陆She ...

  7. turtle库元素语法分析

    一.turtle原理理解: turtle库是Python中一个有趣的图形绘制函数库.原名(海龟),我们想象一只海龟,位于显示器上窗体的正中心,在画布上游走,它游走的轨迹就形成了绘制的图形. 对于小海龟 ...

  8. C# 范型约束 new() 你必须要知道的事

    C# 范型约束 new() 你必须要知道的事 注意:本文不会讲范型如何使用,关于范型的概念和范型约束的使用请移步谷歌. 本文要讲的是关于范型约束无参构造函数 new 的一些底层细节和注意事项.写这篇文 ...

  9. k8s-获取kuboardtoken

    master节点执行命令 echo $(kubectl -n kube-system get secret $(kubectl -n kube-system get secret | grep kub ...

  10. rabbitmq之后台管理和用户设置

    前言 前面介绍了erlang环境的安装和rabbitmq环境安装,接下来介绍rabbitmq的web管理和用户设置. 启用后台管理插件 通过后台管理插件我们可以动态监控mq的流量,创建用户,队列等. ...