㉿㉿㉿c++模板的初阶(通俗易懂简化版)㉿㉿㉿

简介: ㉿㉿㉿c++模板的初阶(通俗易懂简化版)㉿㉿㉿

首先模版就相当于一个模具,可以用它模造出很多实例化的物,形象的比如:

就像是用同一个模具,当放进去的蛋糕原料不同,那么生成的蛋糕种类也不同一样,这里形象的把类型比作原料;而又可以充分解释类模版和模版类的关系,也就是模具和蛋糕的关系,利用类模版具体化出模版类来(以下会涉及讲到)。

以下是对模版的分类:

下面展开来讲:

一·函数模版:
简述:首先与类型无关,在使用的时候才会被实参化,根据实参类型而产生特定的类型,有一个匹配的概念。

模版格式:template +返回值类型 函数名(参数列表){}

typename是用来定义模板参数关键字,也可以使用class但不能使用struct代替 class。

下面用代码写一个简单的函数模版:

template
void Swap( T& left, T& right)
{
T temp = left;
left = right;
right = temp;
}
利用模版实现了一个简单的不同类型之间的交换。

1.函数模版实例化:
①模版实例化概念:即编译器调用相应的函数;而如果类型不同,每次调用的不会是同一个函数。

②当传参给模版,它会进行参数匹配,完成实例化出 实函数然后在调用,这两个不是同一个。

函数模版实例化分为 显示实例化和隐式实例化,下面展开讲解:

1.1显示实例化:
即在函数名后的<>中指定模板参数的实际类型。

代码展示:

template
T Add(const T& left, const T& right)
{
return left + right;
}

int main()
{
int a = 10;
double b = 20.0;
Add(a, b);
return 0;
}
1.2隐式实例化:
即编译器自己进行的内部参数匹配,生成对应函数:如:

template
T 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);
Add(d1, d2);//这里编译器就会自己匹配函数,而如果两个数类型不同,而且类型T只有一个,那么就需要强转【Add(a, (int)d)】或者显示实例化,否则报错
1.3函数模版匹配原则:
①如果有一个与函数模版同名的非模版函数(有明确的类型),而模版传参时候恰好是同类型,这时就无需匹配,直接调用已存在的那个同名非模版函数。

如:

int Add(int left, int right)
{
return left + right;
}

template
T Add(T left, T right)
{
return left + right;
}

void Test()
{
Add(1, 2);
Add(1, 2);
}

②如果有两个可以进行匹配的参数,那么就不用像一个的那样强转了。

int Add(int left, int right)
{
return left + right;

}

template
T1 Add(T1 left, T2 right)
{
return left + right;
}

void Test()
{
Add(1, 2); //这里可以直接调用现成的
Add(1, 2.0); //编译器自己匹配

}

③模板函数不允许自动类型转换,但普通函数可以进行自动类型转换。

如:

template
void fun2(T a, T b)
{ std::cout<<<< std::endl;
}

void funl(double a,double b)
{std::cout<< a<<<< b<< std::endl;
}
void test(){
fun1(10.0,20);//普通函数可以自己强转
fun2(10.0,20);//由于只有一个T,故不能要么自己强转,要么T1,T2。
}

二·类模版:
1.1类模版的定义格式:
template

class 类模板名 {

};

类模版根据传参匹配生成初始化对应的类。

1.2类模版的实例化:
类模板实例化与函数模板实例化不同,类模板实例化需要在类模板名字后跟<>,然后将实例化的 类型放在<>中即可,类模板名字不是真正的类,而实例化的结果才是真正的类。

简单来说对于栈这个类模版 stack只是它的类名,而要进行实例化匹配,它的模版类类型应该是stack<类型>+对象名字。

如:

// Stack是类名,Stack才是类型
Stack st1;
Stack st2;
下面用一个栈的例子来说明一下类模版:

template
class stack {
public:
stack(int count = 4)
:_array(new T[count])
, _top(0)
, _capacity(count) {

}
void push(const T& n);
~stack(){
    delete[]_array;
    _array = nullptr;
    _capacity = _top = 0;
}

private:
T* _array;
size_t _top;
int _capacity;
};

//在类内对类模版成员函数声明,类外定义,需要声明模版,以及类模版的类。
template
void stack::push(const T&n) {
if (_capacity== _top) {
T p = new T[_capacity 2];//这里c++不能像c一样用realloc扩容,只能new新空间,拷贝,再赋给_array。
memcpy(p, _array, _top);
delete[]_array;
_array = p;
_capacity = _capacity * 2;

  }
_array[_top] = n;
_top++;

}

int main() {
stack(2);//机制:类模版根据参数类型识别出要模出哪种类型的stack,利用类的类型转换,去调用类模版的相应参数成员函数,拿它去初始化临时对象,再拷给最后的对象。
stack('a');

return 0;

}*/

但是如果要是声明,定义“大分家”是行不通的,比如声明在.h:而定义在.cpp;这样就会链接错误引起报错。 因此一般不建议这样做。

相关文章
|
4月前
|
缓存 算法 程序员
C++STL底层原理:探秘标准模板库的内部机制
🌟蒋星熠Jaxonic带你深入STL底层:从容器内存管理到红黑树、哈希表,剖析迭代器、算法与分配器核心机制,揭秘C++标准库的高效设计哲学与性能优化实践。
C++STL底层原理:探秘标准模板库的内部机制
|
存储 算法 C++
C++ STL 初探:打开标准模板库的大门
C++ STL 初探:打开标准模板库的大门
315 10
|
编译器 C++
【C++】——初识模板
【C++】——初识模板
【C++】——初识模板
|
8月前
|
存储 算法 安全
c++模板进阶操作——非类型模板参数、模板的特化以及模板的分离编译
在 C++ 中,仿函数(Functor)是指重载了函数调用运算符()的对象。仿函数可以像普通函数一样被调用,但它们实际上是对象,可以携带状态并具有更多功能。与普通函数相比,仿函数具有更强的灵活性和可扩展性。仿函数通常通过定义一个包含operator()的类来实现。public:// 重载函数调用运算符Add add;// 创建 Add 类的对象// 使用仿函数return 0;
276 0
|
8月前
|
人工智能 机器人 编译器
c++模板初阶----函数模板与类模板
class 类模板名private://类内成员声明class Apublic:A(T val):a(val){}private:T a;return 0;运行结果:注意:类模板中的成员函数若是放在类外定义时,需要加模板参数列表。return 0;
226 0
|
11月前
|
编译器 C++
模板(C++)
本内容主要讲解了C++中的函数模板与类模板。函数模板是一个与类型无关的函数家族,使用时根据实参类型生成特定版本,其定义可用`typename`或`class`作为关键字。函数模板实例化分为隐式和显式,前者由编译器推导类型,后者手动指定类型。同时,非模板函数优先于同名模板函数调用,且模板函数不支持自动类型转换。类模板则通过在类名后加`&lt;&gt;`指定类型实例化,生成具体类。最后,语录鼓励大家继续努力,技术不断进步!
|
12月前
|
安全 C++
【c++】模板详解(2)
本文深入探讨了C++模板的高级特性,包括非类型模板参数、模板特化和模板分离编译。通过具体代码示例,详细讲解了非类型参数的应用场景及其限制,函数模板和类模板的特化方式,以及分离编译时可能出现的链接错误及解决方案。最后总结了模板的优点如提高代码复用性和类型安全,以及缺点如增加编译时间和代码复杂度。通过本文的学习,读者可以进一步加深对C++模板的理解并灵活应用于实际编程中。
194 0
|
存储 安全 算法
深入理解C++模板编程:从基础到进阶
在C++编程中,模板是实现泛型编程的关键工具。模板使得代码能够适用于不同的数据类型,极大地提升了代码复用性、灵活性和可维护性。本文将深入探讨模板编程的基础知识,包括函数模板和类模板的定义、使用、以及它们的实例化和匹配规则。
|
安全 编译器 C++
【C++11】可变模板参数详解
本文详细介绍了C++11引入的可变模板参数,这是一种允许模板接受任意数量和类型参数的强大工具。文章从基本概念入手,讲解了可变模板参数的语法、参数包的展开方法,以及如何结合递归调用、折叠表达式等技术实现高效编程。通过具体示例,如打印任意数量参数、类型安全的`printf`替代方案等,展示了其在实际开发中的应用。最后,文章讨论了性能优化策略和常见问题,帮助读者更好地理解和使用这一高级C++特性。
448 4
|
算法 编译器 C++
【C++】模板详细讲解(含反向迭代器)
C++模板是泛型编程的核心,允许编写与类型无关的代码,提高代码复用性和灵活性。模板分为函数模板和类模板,支持隐式和显式实例化,以及特化(全特化和偏特化)。C++标准库广泛使用模板,如容器、迭代器、算法和函数对象等,以支持高效、灵活的编程。反向迭代器通过对正向迭代器的封装,实现了逆序遍历的功能。
225 3