IT数码 购物 网址 头条 软件 日历 阅读 图书馆
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
图片批量下载器
↓批量下载图片,美女图库↓
图片自动播放器
↓图片自动播放器↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁
 
   -> C++知识库 -> 【C++】类型萃取 -> 正文阅读

[C++知识库]【C++】类型萃取

一、类型萃取

笔者目前以这种使用场景来解释什么是类型萃取,为什么需要类型萃取?看代码会更加明了。
Demo.h

#ifndef DEMO_H
#define DEMO_H

//为什么需要萃取器?

//自设计的模板Object<_Ty>类型
template<class _Ty>
class Object
{
private:
	_Ty value;
public:
	Object(_Ty val = 0) : value(val) {}
	~Object() {}

	const _Ty Value()const
	{
		return value;
	}
	_Ty Value()
	{
		return value;
	}
};

//自设计模板容器Container<_Container>类型
template<class _Container>
class Container
{
private:
	_Container c;
public:
	Container(_Container x = 0) : c(x)
	{}
	
	//例如Container<Object<int>> con;
	//con容器中存放的是Object<int>类型的对象
	//此时通过con.getObjectValue()得到存放对象的方法

	返回值类型 ? ? getObjectValue()
	{
		//此时并不知道容器中对象的成员方法的返回值类型
		return c.Value();
	}
};
#endif

通过这个场景可以看出,我们非常有必要知道c.Value()的返回值类型,若是在Object类中,我们很容易知道返回值类型就是_Ty,但是经过一次封装后,通过封装的Container类是不能直接去获得存储在其中的对象的成员中的类型。那么类型萃取就诞生了。(不仅仅局限于这个原因,更重要C++追求效率的缘故,这个后面再说)

那我们加上类型萃取器(类)后:
Demo.h

#ifndef DEMO_H
#define DEMO_H

//为什么需要萃取器?

//自设计的模板类型
template<class _Ty>
class Object
{
private:
	_Ty value;
public:
	//第一步:对需要萃取的类型进行类型重命名
	//这里需要_Ty的类型
	typedef _Ty value_type;		//value值类型

	Object(_Ty val = 0) : value(val) {}
	~Object() {}

	const _Ty Value()const
	{
		return value;
	}
	_Ty Value()
	{
		return value;
	}
};

//Object类的类型萃取器:
/* 
 * 当使用Object<int>类型推演时 
 * 该类中将Object<int>::value_type进行类型重命名value_type
 * 最终在其他地方使用Object_traits<_con>
*/
template<class _Object>
struct Object_traits
{
	typedef typename _Object::value_type value_type;
	//... 需要什么类型就typedef 什么就行
};

//自设计模板容器类型
template<class _Container>
class Container
{
private:
	_Container c;
public:
	Container(_Container x = 0) : c(x)
	{}
	//typename说明value_type是Object_traits中的一个类型
	typename Object_traits<_Container>::value_type getObjectValue()
	{
		return c.Value();
	}
};
#endif

main.cpp

#include <iostream>
#include "Demo.h"
using namespace std;
int main()
{
	Object<int> iobj(10);
	Container<Object<int>> icon(iobj);
	cout << icon.getObjectValue() << endl;

	Object<double> dobj(3.14);
	Container<Object<double>> dcon(dobj);
	cout << dcon.getObjectValue() << endl;

	return 0;
}

运行结果:
在这里插入图片描述
类型萃取时机总结:

  • 模板类A作为模板类B的模板参数,并且模板B类中需要使用模板类A中的模板参数类型

二、通过类型萃取实现函数重载

通过这几种组合起来,我们能达到早绑定的效果:在编译时期就确定函数的调用时机。

Demo.h

#ifndef DEMO_H
#define DEMO_H
#include <iostream>
using namespace std;

//自设计的模板类型
template<class _Ty>
class Object
{
private:
	_Ty value;
public:
	//第一步:对需要萃取的类型进行类型重命名
	//这里需要_Ty的类型
	typedef _Ty value_type;		//value值类型

	Object(_Ty val = 0) : value(val) {}
	~Object() {}

	const _Ty Value()const
	{
		return value;
	}
	_Ty Value()
	{
		return value;
	}
};

//Object类的类型萃取器:
/* 
 * 当使用Object<int>类型推演时 
 * 该类中将Object<int>::value_type进行类型重命名value_type
 * 最终在其他地方使用Object_traits<_con>
*/
template<class _Object>
struct Object_traits
{
	typedef typename _Object::value_type value_type;
	//... 需要什么类型就typedef 什么就行
};


//模板函数:可以作为其他(模板)函数重载的条件
template<class _OBj>
typename Object_traits<_OBj>::value_type dif_Condition(const _OBj&)
{
	//返回该类的临时对象
	return typename Object_traits<_OBj>::value_type();
}

//自设计模板容器类型
template<class _Container>
class Container
{
private:
	_Container c;
public:
	Container(_Container x = 0) : c(x)
	{}
	//typename说明value_type是Object_traits中的一个类型
	typename Object_traits<_Container>::value_type getObjectValue()
	{
		return c.Value();
	}
private:
	//两个重载的函数
	void __fun(int) 
	{
		cout << " int "<< endl;
	}
	void __fun(double)
	{
		cout << " double " << endl;
	}
public:
	void fun()
	{
		//通过dif_condition的返回值(对象)的类型进行不同函数的调用
		__fun(dif_Condition(c));
	}
};
#endif

main.cpp

#include <iostream>
#include "Demo.h"
using namespace std;
int main()
{
	Object<int> iobj(10);
	Container<Object<int>> icon(iobj);
	cout << icon.getObjectValue() << endl;
	icon.fun();

	Object<double> dobj(3.14);
	Container<Object<double>> dcon(dobj);
	cout << dcon.getObjectValue() << endl;
	dcon.fun();
	return 0;
}

测试结果:
在这里插入图片描述

本节主要为了充分理解类型粗萃取的原理,后面将会对STL中的迭代器进行论述。

  C++知识库 最新文章
【C++】友元、嵌套类、异常、RTTI、类型转换
通讯录的思路与实现(C语言)
C++PrimerPlus 第七章 函数-C++的编程模块(
Problem C: 算法9-9~9-12:平衡二叉树的基本
MSVC C++ UTF-8编程
C++进阶 多态原理
简单string类c++实现
我的年度总结
【C语言】以深厚地基筑伟岸高楼-基础篇(六
c语言常见错误合集
上一篇文章      下一篇文章      查看所有文章
加:2022-03-12 17:14:26  更:2022-03-12 17:15:19 
 
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁

360图书馆 购物 三丰科技 阅读网 日历 万年历 2025年1日历 -2025/1/10 16:28:26-

图片自动播放器
↓图片自动播放器↓
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
图片批量下载器
↓批量下载图片,美女图库↓
  网站联系: qq:121756557 email:121756557@qq.com  IT数码