app网站建设可行性分析,青海高端网站建设价格,轻应用网站模板,我要建网站需要什么文章目录 C模版1、泛型编程2、函数模版2.1、函数模版概念2.2、函数模版格式2.3、函数模版原理2.4、函数模版的实例化2.5、模板参数的匹配原则 3、类模版3.1、类模版概念3.2、类模版格式3.3、类模板的实例化 C模版
1、泛型编程 泛型编程#xff08;Generic Programming#x… 文章目录 C模版1、泛型编程2、函数模版2.1、函数模版概念2.2、函数模版格式2.3、函数模版原理2.4、函数模版的实例化2.5、模板参数的匹配原则 3、类模版3.1、类模版概念3.2、类模版格式3.3、类模板的实例化 C模版
1、泛型编程 泛型编程Generic Programming是一种编程范式它强调编写可以在不同数据类型上工作的通用代码。泛型编程的目标是编写更具通用性和复用性的代码以减少代码的重复编写并提高代码的可维护性。 泛型编程的主要特点包括 通用性 泛型编程的代码不依赖于特定的数据类型而是能够在多种不同数据类型上工作。这使得代码更具通用性。类型参数化 泛型编程通过参数化类型允许在代码中使用参数化类型的变量、函数或类。这意味着你可以编写一次通用代码然后在不同的上下文中使用不同的类型。类型安全 泛型编程强调类型安全因为它在编译时进行类型检查避免了运行时的类型错误。代码复用 泛型代码可以在不同的场景中重复使用减少了代码的冗余。 如何实现一个通用的交换函数呢 ? void Swap(int left, int right) {int temp left;left right;right temp;
}void Swap(double left, double right) {double temp left;left right;right temp;
}void Swap(char left, char right) {char temp left;left right;right temp;
}//......
int main() {int a 1, b 2;double c 1.1, d 2.2;Swap(a, b);Swap(c, d);return 0;
}使用函数重载虽然可以实现但是有以下几个不好的地方 重载的函数仅仅是类型不同代码复用率比较低只要有新类型出现时就需要用户自己增加对应的函数 代码的可维护性比较低一个出错可能所有的重载均出错 那能否告诉编译器一个模子让编译器根据不同的类型利用该模子来生成代码呢 答C里面有一种机制 — 模版函数模版和类模版。 泛型编程编写与类型无关的通用代码是代码复用的一种手段。模板是泛型编程的基础。 2、函数模版
2.1、函数模版概念 函数模板Function Template是C中的一种机制允许你编写可以应用于不同数据类型的通用函数。函数模板允许你编写一次通用函数定义然后可以将不同的数据类型作为参数传递给该函数从而自动生成特定数据类型的函数实例。 函数模板的基本概念包括 通用性 函数模板允许你编写通用函数不限于特定数据类型。这使得代码更具通用性和复用性。类型参数 函数模板通过使用类型参数通常用typename或class关键字指定来表示可以应用于不同数据类型的函数。实例化 当你调用函数模板并传递特定数据类型时编译器会根据传递的数据类型生成特定数据类型的函数实例。这个过程称为实例化。类型推导 C编译器通常能够从传递的参数推断出要使用的数据类型从而实例化函数模板。
2.2、函数模版格式 格式templatetypename T1, typename T2,......,typename Tn 举例 单参数函数模版 templatetypename Tvoid Swap(T a, T b) {T tmp a;a b;b tmp;
}int main() {int a 1, b 2;double c 1.1, d 2.2;Swap(a, b);Swap(c, d);return 0;
}多参数函数模版 templatetypename T1, class T2void Swap(T1 a, T2 b) {T1 tmp a;a b;b tmp;
}int main() {int a 1, b 2;double c 1.1, d 2.2;Swap(a, b);Swap(c, d);Swap(a, c);return 0;
}注意typename是用来定义模板参数关键字也可以使用class切记不能使用struct代替class。
2.3、函数模版原理 函数模板是一个蓝图它本身并不是函数是编译器用使用方式产生特定具体类型函数的模具。所以其实模板就是将本来应该我们做的重复的事情交给了编译器。 在编译器编译阶段对于模板函数的使用编译器需要根据传入的实参类型来推演生成对应类型的函数以供调用。比如当用double类型使用函数模板时编译器通过对实参类型的推演将T确定为double类型然后产生一份专门处理double类型的代码对于字符类型也是如此。 注意这三次调用的Swap不是同一个函数 我们vs转到反汇编来看函数地址
2.4、函数模版的实例化 使用不同类型的参数使用函数模板时称为函数模板的实例化。模板参数实例化分为隐式实例化和显式实例化。 隐式实例化让编译器根据实参推演模板参数的实际类型。 templateclass TT Add(const T left, const T right) {return left right;
}int main() {int a1 10, a2 20;double d1 10.0, d2 20.0;Add(a1, a2);//推导出a1a2是int类型Add(d1, d2);//推导出d1d2是double类型/*该语句不能通过编译因为在编译期间当编译器看到该实例化时需要推演其实参类型通过实参a1将T推演为int通过实参d1将T推演为double类型但模板参数列表中只有一个T编译器无法确定此处到底该将T确定为int 或者 double类型而报错注意在模板中编译器一般不会进行类型转换操作因为一旦转化出问题编译器就需要背黑锅Add(a1, d1);*/// 此时有两种处理方式1. 用户自己来强制转化 2. 使用显式实例化//1. 用户自己来强制转化 隐式实例化int a 1;double d 2.1;Add(a, (int) d);//2. 使用显式实例化int b 1;double c 2.1;Addint(b, c);//这里c就被强转化为int类型return 0;
}显式实例化在函数名后的中指定模板参数的实际类型。 templateclass TT Add(const T left, const T right) {return left right;
}int main() {int b 1;double c 2.1;Adddouble(b, c);//这里b就被强转化为double类型return 0;
}函数参数里如果没有模版则需要显式实例化。 templateclass TT Add(const int left, const int right) {T sum left right;return sum;
}int main() {int b 1;double c 2.1;int a Addint(b, c);//这里b就被强转化为double类型return 0;
}如果类型不匹配编译器会尝试进行隐式类型转换如果无法转换成功编译器将会报错。
2.5、模板参数的匹配原则 一个非模板函数可以和一个同名的函数模板同时存在而且该函数模板还可以被实例化为这个非模板函数。 // 专门处理int的加法函数
int Add(int left, int right) {return left right;
}// 通用加法函数
templateclass T
T Add(T left, T right) {return left right;
}int main() {Add(1, 2); // 与非模板函数匹配编译器不需要特化Addint(1, 2); // 调用编译器特化的Add版本return 0;
}对于非模板函数和同名函数模板如果其他条件都相同在调动时会优先调用非模板函数而不会从该模板产生出一个实例。如果模板可以产生一个具有更好匹配的函数 那么将选择模板。 // 专门处理int的加法函数
int Add(int left, int right) {return left right;
}// 通用加法函数
templateclass T1, class T2
T1 Add(T1 left, T2 right) {return left right;
}int main() {Add(1, 2); // 与非函数模板类型完全匹配不需要函数模板实例化Add(1, 2.0); // 模板函数可以生成更加匹配的版本编译器根据实参生成更加匹配的Add函数return 0;
}模板函数不允许自动类型转换但普通函数可以进行自动类型转换。 3、类模版
3.1、类模版概念 类模板Class Template是C中的一种机制允许你定义通用类其中的数据成员和成员函数可以适用于不同的数据类型。类模板允许你编写一次通用类定义然后可以将不同的数据类型作为模板参数传递给该类从而自动生成特定数据类型的类实例。 类模板的基本概念包括 通用性 类模板允许你编写通用类不限于特定数据类型。这使得代码更具通用性和复用性。类型参数 类模板通过使用类型参数通常用typename或class关键字指定来表示可以应用于不同数据类型的类。实例化 当你创建类模板的对象时编译器会根据传递的数据类型生成特定数据类型的类实例。这个过程称为实例化。类型推导 C编译器通常能够从传递的参数推断出要使用的数据类型从而实例化类模板。
3.2、类模版格式 格式 templateclass T1, class T2, ..., class Tn
class 类模板名
{// 类内成员定义
};举例 // 动态顺序表
// 注意Vector不是具体的类是编译器根据被实例化的类型生成具体类的模具
templateclass T
class Vector {
public :Vector(size_t capacity 10): _pData(new T[capacity]), _size(0), _capacity(capacity) {}// 使用析构函数演示在类中声明在类外定义。~Vector();void PushBack(const T data);void PopBack();// ...size_t Size() { return _size; }T operator[](size_t pos) {assert(pos _size);return _pData[pos];}private:T *_pData;size_t _size;size_t _capacity;
};// 注意类模板中函数放在类外进行定义时需要加模板参数列表
templateclass T
VectorT::~Vector() {if (_pData)delete[] _pData;_size _capacity 0;
}class Date {
public:Date() : _year(1), _month(1), _day(1) {}private:int _year;int _month;int _day;
};注意Vector不是具体的类是编译器根据被实例化的类型生成具体类的模具。
3.3、类模板的实例化 类模板实例化与函数模板实例化不同类模板实例化需要在类模板名字后跟然后将实例化的类型放在中即可类模板名字不是真正的类而实例化的结果才是真正的类。 // Vector是类名 Vector类型才是类型
int main() {Vectorint it;VectorDate dt;return 0;
}OKOKC模版就到这里。如果你对Linux和C也感兴趣的话可以看看我的主页哦。下面是我的github主页里面记录了我的学习代码和leetcode的一些题的题解有兴趣的可以看看。
Xpccccc的github主页