C++语言基础(19)-模板的显式具体化
应用背景:
例如有下面的函数模板,它用来获取两个变量中较大的一个:
- template<class T> const T& Max(const T& a, const T& b){
- return a > b ? a : b;
- }
请读者注意a > b
这条语句,>
能够用来比较 int、float、char 等基本类型数据的大小,但是却不能用来比较结构体变量、对象以及数组的大小,因为我们并没有针对结构体、类和数组重载>
。
另外,该函数模板虽然可以用于指针,但比较的是地址大小,而不是指针指向的数据,所以也没有现实的意义。
让模板能够针对某种具体的类型使用不同的算法(函数体或类体不同),这在 C++ 中是可以做到的,这种技术称为模板的显示具体化(Explicit Specialization)。
一.函数模板的显式具体化
- #include <iostream>
- #include <string>
- using namespace std;
- typedef struct {
- string name;
- int age;
- float score;
- } STU;
- template<typename T>
- const T& Max(const T &a, const T &b);
- template<>
- const STU& Max<STU>(const STU &a, const STU &b);
- ostream & operator<<(ostream &out, const STU &stu);
- int main() {
- int a = ;
- int b = ;
- cout<<Max(a,b)<<endl;
- STU stu1 = {"Jack",,95.5};
- STU stu2 = {"Mike",,90.0};
- cout<<Max(stu1,stu2)<<endl;
- return ;
- }
- template<typename T>
- const T& Max(const T &a, const T &b) {
- return a > b ? a : b;
- }
- template<>
- const STU& Max<STU>(const STU &a, const STU &b) {
- return a.score > b.score ? a : b;
- }
- ostream & operator<<(ostream &out, const STU &stu) {
- out<<stu.name<<", "<<stu.age<<", "<<stu.score;
- return out;
- }
运行结果:
Jack,16,95.5
语法格式为:
- template<> const STU& Max(const STU& a, const STU& b);
二.类模板显式具体化
- #include <iostream>
- #include <string>
- using namespace std;
- template<typename T1, typename T2>
- class Point {
- public:
- Point(T1 x, T2 y):m_x(x), m_y(y){}
- public:
- T1 getX() const {return m_x;}
- void setX(T1 x) {m_x = x;}
- T2 getY() const {return m_y;}
- void setY(T2 y) {m_y = y;}
- void display() const;
- private:
- T1 m_x;
- T2 m_y;
- };
- // 这里要带上模板头
- template<typename T1, typename T2>
- void Point<T1,T2>::display() const {
- cout<<"x="<<m_x<<", y="<<m_y<<endl;
- }
- // 类模板显式具体化(针对字符串类型的显式具体化)
- template<>
- class Point<char *, char *> {
- public:
- Point(char *x, char *y):m_x(x), m_y(y){}
- public:
- char* getX() const {return m_x;}
- void setX(char *x) {m_x = x;}
- char *getY() const {return m_y;}
- void setY(char *y) {m_y = y;}
- void display() const;
- private:
- char *m_x;
- char *m_y;
- };
- // 注意!这里不能带模板头template<>
- void Point<char*, char*>::display() const {
- cout<<"x="<<m_x<<", y="<<m_y<<endl;
- }
- int main() {
- (new Point<int,int>(,))->display();
- (new Point<int, char*>(,"jack"))->display();
- (new Point<char*,char*>("java","android"))->display();
- return ;
- }
运行结果:
x =10, y = 20
x = 20, y = jack
x = java, y = android
需要注意的是:在类模板的具体化中,成员方法的实例化是不能带模板头template<>的。
三.部分显式具体化
- #include <iostream>
- using namespace std;
- // 类模板
- template<typename T1, typename T2>
- class Point {
- public:
- Point(T1 x, T2 y):m_x(x), m_y(y){}
- public:
- T1 getX() const {return m_x;}
- void setX(T1 x){m_x = x;}
- T2 getY() const {return m_y;}
- void setY(T2 y){m_y = y;}
- void display() const;
- private:
- T1 m_x;
- T2 m_y;
- };
- template<typename T1, typename T2>
- void Point<T1,T2>::display() const {
- cout<<"x="<<m_x<<", y="<<m_y<<endl;
- }
- template<typename T2>
- class Point<char*, T2> {
- public:
- Point(char *x, T2 y):m_x(x), m_y(y){}
- public:
- char *getX() const {return m_x;}
- void setX(char *x){m_x = x;}
- T2 getY() const {return m_y;}
- void setY(T2 y){m_y = y;}
- void display() const;
- private:
- char *m_x;
- T2 m_y;
- };
- // 部分显式具体化还是需要加上模板头
- template<typename T2>
- void Point<char*,T2>::display() const {
- cout<<"x="<<m_x<<" | y="<<m_y<<endl;
- }
- int main() {
- (new Point<int,int>(,))->display();
- (new Point<char*,int>("jack",))->display();
- (new Point<char*,char*>("java","android"))->display();
- return ;
- }
运行结果:
x = 10, y = 20
x = jack | y = 10
x = java | y = android
注意:
部分显式具体化只能用于类模板,不能用于函数模板
C++语言基础(19)-模板的显式具体化的更多相关文章
- C++语言基础(18)-模板
Java中的泛型编程可以极大的提升编程的效率,比如在android中查找一个控件的ID:标准写法为: TextView tv_text = (TextView)findViewById(R.id.tv ...
- C++语言基础(20)-模板的非类型参数
一.在函数模板中使用非类型参数 #include <iostream> using namespace std; template<class T> void Swap(T & ...
- Java入门 - 语言基础 - 19.方法
原文地址:http://www.work100.net/training/java-method.html 更多教程:光束云 - 免费课程 方法 序号 文内章节 视频 1 概述 2 方法的定义 3 方 ...
- C语言基础(19)-结构体,联合体,枚举和typedef
一.结构体 1.1 结构体struct定义及初始化 #include <stdio.h> // 这个头文件在系统目录下 #include <stdlib.h> // 使用了sy ...
- day02<Java语言基础+>
Java语言基础(常量的概述和使用) Java语言基础(进制概述和二,八,十六进制图解) Java语言基础(不同进制数据的表现形式) Java语言基础(任意进制到十进制的转换图解) Java语言基础( ...
- 02 java语言基础
常量:字面值常量(字符串,字符,整数,小数,布尔,null),自定义常量,''这个不是字符常量,""这个是字符串常量 进制: 02.01_Java语言基础(常量的概述和使用) A: ...
- C++基础--函数模板
函数模板是通用的函数描述,其使用泛型来定义函数.其实就是有些操作,如果撇开具体的变量的数据类型,其操作是一样的如果我们将这些操作写成一个模板,在调用不同变量的时候就设定好变量类型就可了,后续的操作基本 ...
- 并发编程 19—— 显式的Conditon 对象
Java并发编程实践 目录 并发编程 01—— ThreadLocal 并发编程 02—— ConcurrentHashMap 并发编程 03—— 阻塞队列和生产者-消费者模式 并发编程 04—— 闭 ...
- C++学习笔记36 (模板的细节明确template specialization)和显式实例(template instantiation)
C++有时模板很可能无法处理某些类型的. 例如: #include <iostream> using namespace std; class man{ private: string n ...
随机推荐
- mysql 获取当前日期及格式化(转)
MySQL 获取当前日期及日期格式 获取系统日期: NOW() 格式化日期: DATE_FORMAT(date, format) 注: date:时间字段,format:日期格式 select now ...
- 【二分法】【尺取法】bzoj2348 [Baltic 2011]Plagiarism
一开始以为死于精度……调了半天发现死于long long…… 一.二分法: #include<cstdio> #include<cstring> #include<alg ...
- 【进制转换】codevs 1474 十进制转m进制
#include<cstdio> using namespace std; ],en; int main() { scanf("%d%d",&n,&m) ...
- 【左偏树】BZOJ2809-[APIO2012]dispatching
[题目大意] 在一个忍者的帮派里,一些忍者们被选中派遣给顾客,然后依据自己的工作获取报偿.在这个帮派里,有一名忍者被称之为 Master.除了 Master以外,每名忍者都有且仅有一个上级.为保密,同 ...
- 1.1(Mybatis学习笔记)初识Mybatis
一.Mybatis下载与使用 下载地址:https://github.com/mybatis/mybatis-3/releases 下载后解压目录: 需要将lib下的jar包和mybatid-x-x- ...
- @RequestParam注解的使用
自SpringMVC4.2之后,RequestParam内部有4个参数: 1.String name; 2.String value; 3.boolean required; 4.String def ...
- Problem A: 零起点学算法16——鸡兔同笼
#include<stdio.h> int main() { int n,m,a,b; while(scanf("%d %d",&n,&m)!=EOF) ...
- BUG:Yii登录时 101 net::ERR_CONNECTION_RESET
Bug描述:YII web入口登录,无法登录一直等待,最终重定向 原因:设置的默认路由DefauRoute中的控制器中有错误,导致无法跳转找指定的路由规则 解决方案:这就多亏了SourceTree了, ...
- 网络采集软件核心技术剖析系列(3)---如何使用C#语言下载博文中的全部图片到本地并可以离线浏览
一 本系列随笔概览及产生的背景 本系列开篇受到大家的热烈欢迎,这对博主是莫大的鼓励,此为本系列第三篇,希望大家继续支持,为我继续写作提供动力. 自己开发的豆约翰博客备份专家软件工具问世3年多以来,深受 ...
- RequireJs 入门
官网:http://www.requirejs.cn/ 使用方法: 1.引入require.js 可以在底部引入: <script type="text/javascript" ...