C++非类型模板参数

一、非类型模板参数

简单点说就是在定义模板时,模板的类型参数不是一个泛型的 T,而是一个具体的类型。这种比较多的是在比如定就义容器类模板时,指定容器的容量大小。如下 SStack 的定义。

template<typename T, std::size_t MaxSize>
class SStack{
private:
    std::array<T, MaxSize> elems;
    std::size_t numElems;
public:
    SStack();
    void push(const T& t);
    void pop();
    const T& top() const;
    bool empty() const {
        return numElems == 0;
    }
    std::size_t size() const {
        return numElems;
    }
};

template<typename T, std::size_t MaxSize>
SStack<T, MaxSize>::SStack() {}

template<typename T, std::size_t MaxSize>
void SStack<T, MaxSize>::push(const T &t) {
    assert(numElems < MaxSize);
    elems[numElems] = t;
    ++numElems;
}

template<typename T, std::size_t MaxSize>
void SStack<T, MaxSize>::pop() {
    assert(!elems.empty());
    --numElems;
}

template<typename T, std::size_t MaxSize>
const T& SStack<T, MaxSize>::top() const{
    assert(!elems.empty());
    return elems[numElems-1];
}

下面是使用代码

SStack<float, 30> sStack30;
SStack<float, 50> sStack50;

看起来还是很简单的,谁说不是呢?

二、非类型模板参数的限制

非类型模板参数并不是所有类型都可以的,是有一定限制的:

  1. 只能是整型常量包括枚举,比如,如果是个 std::string 则会报以下错误信息。


    image.png
  2. 指向 objects/function/members 的指针或者引用。但注意,指针所指向的对象不能是字符串常量、局部变量或者数据成员及其子类对象。而且,这里还有版本的差异。在 C++11 中,要求指针所指的对象必须是 extern 的,而在 C++ 14 则要求必须 extern 或者文件内的全局变量,而在 C++ 17 后,这些限制都放宽了。

  3. objects/function 的左值引用,或者 nullptr。

三、使用 auto 关键字使非类型模板参数的类型更加泛化

使用 auto 关键字作为非类型模板参数时的定义

template<typename T, auto MaxSize>
class AutoStack{
private:
    std::array<T, MaxSize> elems;
    using size_type = decltype(MaxSize);
    size_type numElems;
public:
    AutoStack();
    void push(const T& t);
    void pop();
    const T& top() const;
    bool empty() const {
        return numElems == 0;
    }
    auto size() const {
        return numElems;
    }
};

template<typename T, auto MaxSize>
AutoStack<T, MaxSize>::AutoStack() {}

template<typename T, auto MaxSize>
void AutoStack<T, MaxSize>::push(const T &t) {
    assert(numElems < MaxSize);
    elems[numElems] = t;
    ++numElems;
}

template<typename T, auto MaxSize>
void AutoStack<T, MaxSize>::pop() {
    assert(!elems.empty());
    --numElems;
}

template<typename T, auto MaxSize>
const T& AutoStack<T, MaxSize>::top() const{
    assert(!elems.empty());
    return elems[numElems-1];
}

上面有两个比较有意思的地方,一个是可以通过 decltype 返回 auto 的实际类型从而来定义 numElems 属性。另一个是 size() 方法的返回属性可以用 auto 关键字来让编译器自动推导。真的是妙啊。

四、总结

  1. 通过非类型模板参数,我们可以知道模板的参数类型不一定非得是类型,也可以是数值。只不过这个数值是有一定限制的,具体限制看第 2 点。
  2. 不可以将浮点型或者 class 类型的对象用于非类型模板参数。使用指向字符串常量,临时变量和子对象的指针或者引用也有一些限制。
  3. 使用 auto 关键字,可以使非类型模板参数的类型更加泛化。
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,233评论 6 495
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,357评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,831评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,313评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,417评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,470评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,482评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,265评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,708评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,997评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,176评论 1 342
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,827评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,503评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,150评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,391评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,034评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,063评论 2 352

推荐阅读更多精彩内容

  • 非类型的类模板参数 使用该模板需要同时指定类型和个数 非类型模板参数也可以指定默认值 非类型的函数模板参数 这类模...
    奇点创客阅读 171评论 2 2
  • 前面的例子中,我们分别使用了类型和模板作为类模板的参数。除此之外,模板还支持非类型模板参数。 如下用数组实现Sta...
    MagicBowen阅读 4,940评论 2 0
  • 非类型模板参看,顾名思义,模板参数不限定于类型,普通值也可作为模板参数。在基于类型的模板中,模板实例化时所依赖的是...
    cp3_1dbc阅读 1,328评论 0 0
  • 非类型参数: non-type template argument, 支持的类型定义an integral typ...
    极光火狐狸CPP阅读 233评论 0 0
  • 1. 类型别名作为成员(P243 除了定义数据和函数成员之外,类还可以自定义某种类型在类中的别名。由类定义的类型名...
    何幻阅读 2,004评论 0 1