【C++第三阶段】模板类模板通用数组实现案例

以下内容仅为当前认识,可能有不足之处,欢迎讨论!


文章目录

  • 模板
    • 怎么使用模板
    • 函数模板注意事项
    • 普通函数与函数模板的区别
    • 普通函数与函数模板调用规则
    • 函数模板限制
  • 类模板
    • 类模板语法
    • 类模板与函数模板区别
    • 类模板中成员函数创建时机
    • 类模板对象做函数参数
      • 打印类别
    • 类模板与继承
    • 类模板成员函数类外实现
    • 类模板分文件编写
    • 类模板与友元
    • 类模板案例-通用数组实现


本教程针对泛型编程和STL做技术讲解,探讨更深层的使用。

模板

什么是模板,模板的目的是什么?怎么用模板,优缺点,能用在哪里。

C++提供两种模板机制,函数模板和类模板。

什么是模板?

建立通用的模具,提高复用性。

模板的目的是什么?

提高代码的复用性,同时解决函数重载的问题:不能根据情况指定不同的返回值类型。以及其他函数重载的问题(2024年3月26日20点44分暂时想不到了)

黑马:建立通用函数,其函数返回值类型和形参类型可以不具体指定,用一个虚拟的类型代表。

怎么使用模板

首先,明确模板的语法如下:

template<typename T>
//后写方法

其中

template 是创建模板的声明,说明下面接着的方法用到了模板。

typename是传入不同类型的参数,这个类型是typename。

黑马:typename-表明其后符号是一种数据类型,可以用class类型。

T是通用数据类型,名称可以替换,通常为大写字母。

接着是模板使用具体案例:

#include<iostream>
#include<string>
using namespace std;//以下是模板的简单举例
//说明返回值是temp。
template <typename T>
T MySwap(T& param1, T& param2) {T temp = param1;param1 = param2;param2 = temp;return temp;
}void test0326a() {int a = 10;int b = 20;cout << "now , a = " << a << " , b = " << b << " . " << endl;//隐式调用模板函数MySwap(a, b);cout << "After implicitly utilizing swap func , a= " << a << " , b = " << b << " . " << endl;//显式调用模板函数MySwap<int>(a, b);cout << "After explicitly utilizing swap func , a = " << a << " , b = " << b << " . " << endl;
}int main() {test0326a();system("pause");return 0;
}

可以看到,有两种使用模板方式,隐式及显式。隐式直接传入参数即可,由编译器自行判断参数是何数据类型。显式需要在调用时在函数名与参数列表中用<>说明是何数据类型。

运行结果:

image-20240326231120001

函数模板注意事项

隐式:自动类型推导,必须推导出一致的数据类型T,才可以使用。

显式:模板必须要确定T的数据类型,才可以使用。

注意事项一:必须推导出一致的数据类型T才可以使用。比如下面的将C作为参数传入,识别出两个数据类型不一致。

image-20240326214133982

注意事项二:既然用了函数模板,就需要确定T的数据类型。

像下面的函数,如果隐式地让编译器自动确定数据类型,编译器不知道要确定哪个数据类型。就会报错。所以需要显式地指定数据类型。

image-20240326214501054

image-20240326214454022

所以,解决办法就是显式地指定数据类型。

void test0326c() {print<float>();
}

对应代码:

#include<iostream>
#include<string>
using namespace std;//以下是模板的简单举例
//说明返回值是temp。
template <typename T>
T MySwap(T& param1, T& param2) {T temp = param1;param1 = param2;param2 = temp;return temp;
}template <typename T>
void print() {cout << "print 函数运行." << endl;
}void test0326b() {int a = 10;int b = 20;char c = 'c';cout << "原来的a = " << a << " , b = " << b << " . " << endl;//MySwap(a, c);
}void test0326c() {print<float>();
}int main() {test0326c();system("pause");return 0;
}

运行结果:

image-20240326231048933

一个通用模板排序数组的案例代码:

#include<iostream>
#include<string>
using namespace std;template<typename T>
void MySwap(T &param1 , T &param2){T temp = param1;param1 = param2;param2 = temp;
}template<typename T>
void print(T& array, int number) {//cout << "当前数组为:";for (int i = 0; i < number; i++) {if (array[i] != '\0') {cout << array[i] << "\t";}}cout << endl;
}template <typename T>
void sort(T& array) {int member_number = sizeof(array) / sizeof(array[0]);print(array, member_number);for (int i = 0; i < member_number; i++) {for (int j = i; j < member_number-1; j++) {if (array[i] > array[j + 1]) {MySwap(array[i], array[j + 1]);}}}print(array, member_number);
}int main() {int int_array[] = { 2,9 ,3, 5 , 1 };float float_array[] = { 3.1 , 9.1 , 3.6 , 5.2 };char char_array[] = "fezcda";sort(char_array);sort(int_array);sort(float_array);system("pause");return 0;}

运行结果:

image-20240326230951333

普通函数与函数模板的区别

区别:是否会发生隐式类型转换。

A—普通函数传入参数时,会发生隐式类型转换。

B—a函数模板传入参数,如果是隐式地自动类型推导,不会发生隐式类型转换,会发生报错。

B—b函数模板传入参数,如果是显式地指定类型方式,会发生隐式类型转换。

根据以上三个编号,分别设计代码验证。

#include<iostream>
#include<string>
using namespace std;//设计简单的加法函数与加法重载
int MyAdd(int a, int b) {//如果是引用方式就不可以return a + b;
}template<typename T>
T MyTempAdd(T a, T b) {//如果是引用方式就不可以return a + b;
}int main() {int int_array[] = { 2,9 ,3, 5 , 1 };float float_array[] = { 3.1 , 9.1 , 3.6 , 5.2 };char char_array[] = "fezcda";int a = 10;int b = 13;char c = 'c'; //ASCII = 99cout <<"myadd a+b = " << MyAdd(a, b) << endl;cout << "myadd a+c = " << MyAdd(a, c) << endl;cout << "MyTempAdd a+b = " << MyTempAdd(a, b) << endl;cout << "MyTempAdd a+c = " << MyTempAdd<int>(a, c) << endl;//必须显式指定数据类型,否则会报错。system("pause");return 0;
}

image-20240326230719245

运行结果:

image-20240326230823336

问题:这里就不能再传入引用,如果引用,必须是对应的数据类型,无论是函数还是模板。

普通函数与函数模板调用规则

1.如果函数模板和普通函数都可以实现,优先调用普通函数。

2.可以通过空模板参数列表来强制调用函数模板。

3.函数模板也可以发生重载。

4.如果函数模板可以产生更好的匹配,优先调用函数模板。

1.优先调用普通函数

#include<iostream>
#include<string>
using namespace std;void print( int a) {cout << "调用普通函数" << endl;
}template<typename T>
void print(T a) {cout << "调用函数模板" << endl;
}void test0326e() {int a = 1;print(a);
}int main() {test0326e();system("pause");return 0;
}

通过结果可以看出,优先调用了普通函数。

image-20240327000647436

2.可以通过空模板参数列表强制调用函数模板。

在前面加上空的英文书名号就可以。

更改test0326e()函数中代码为:

void test0326e() {int a = 1;print<>(a);
}

即可优先调用函数模板

image-20240327000801084

3.函数模板也可以发生重载。

参数个数不同,参数类型不同。

#include<iostream>
#include<string>
using namespace std;template<typename T>
void print(T a) {cout << "调用函数模板" << endl;
}template<typename T>
void print(T a, T b) {cout << "调用重载的函数模板" << endl;
}void test0326e() {int a = 1;int b = 2;print<>(a ,b);
}int main() {test0326e();system("pause");return 0;
}

可以看到,调用了重载的函数模板。

image-20240327000930124

4.如果函数模板可以产生更好的匹配,优先调用函数模板。

怎么理解?比如下面代码,传入的是字符,在普通函数中,可以隐式转换数据类型到int,但是函数模板可以不用隐式转换直接到char。所以编译器会执行进函数重载中。

#include<iostream>
#include<string>
using namespace std;void print( int a) {cout << "调用普通函数" << endl;
}template<typename T>
void print(T a) {cout << "调用函数模板" << endl;
}void test0326e() {int a = 1;int b = 2;char c = 'c';print<>(c);
}int main() {test0326e();system("pause");return 0;
}

运行结果:

image-20240327001147736

函数模板限制

普通的数据类型,当然可以运算。

#include<iostream>
#include<string>
using namespace std;template<typename T>
bool MyCompare(T a, T b) {if (a == b) {return true;}else {return false;}
}void test0326b() {int a = 10;int b = 10;bool res = MyCompare(a, b);if (res) {cout << "a == b" << endl;}else {cout << "a != b" << endl;}
}int main() {test0326b();system("pause");return 0;
}

运行结果:

image-20240326235833087

如果是自定义的数据类型,也会出现无法运算的时候。

如果是自定义数据类型,就如果想要走特殊的通道到达对应的函数模板,则需要特殊操作。

#include<iostream>
#include<string>
using namespace std;class Person {
public:Person(int age, string name) {this->p_age = age;this->p_name = name;}public:int p_age;string p_name;
};template<typename T>
bool MyCompare(T a, T b) {if (a == b) {return true;}else {return false;}
}template<> bool MyCompare(Person p1, Person p2) {if (p1.p_age == p2.p_age && p1.p_name == p2.p_name) {return true;}else {return false;}
}void test0326b() {int a = 10;int b = 10;bool res = MyCompare(a, b);if (res) {cout << "a == b" << endl;}else {cout << "a != b" << endl;}
}void test0326d() {Person tom(12, "Cooper");Person Jack(12, "Cooper");bool res = MyCompare(tom, Jack);if (res) {cout << "a == b" << endl;}else {cout << "a != b" << endl;}
}int main() {test0326d();//test0326b();system("pause");return 0;
}

运行结果:

image-20240327000237165

类模板

类模板语法

目的/作用:

建立一个通用类,类中的成员数据类型可以不具体指定,用一个虚拟的类型来代表。

我理解:望名思义,以为是通用的类,类名也是通用的。但其实是类里面的属性或者方法,可以通用。

语法:

template <typename T1 ,typename T2 , typename ,Tn>
class{}

template 声明创建模板

typename 表明后面的符号是一种数据类型,具体是什么数据类型可以自己指定。

T 通用的数据类型,名称可以替换,通常为大写字母。

案例代码:

#include<iostream>
#include<string>
using namespace std;template<typename name_type , typename age_type>
class Person {
public:Person(name_type name, age_type age) :p_name(name), p_age(age) {}public:name_type p_name;age_type p_age;public:void print() {cout << "name = " << p_name << " , age = " << p_age << " . " << endl;}
};void test0327a() {Person <string, int>p("Json", 23);p.print();
}int main() {//test0326e();test0327a();system("pause");return 0;
}

运行结果:

image-20240327194736825

类模板与函数模板区别

区别①类模板不能自动推导类型。

函数模板可以通过传入参数,由编译器判断谁是什么数据类型,而类模板没有这个功能。

区别②类模板可以对类型赋默认值。

如果类中成员属性≥1,可以对成员属性的类型赋默认值。

以下是两点具体代码体现。

void test0327b() {Person p("docker", 99);p.print();
}

区别①会报错:

image-20240327200148738

区别②:

#include<iostream>
#include<string>
using namespace std;template<typename name_type , typename age_type = int>//体现区别2,可以对类型赋默认值。
class Person {
public:Person(name_type name, age_type age) :p_name(name), p_age(age) {}public:name_type p_name;age_type p_age;public:void print() {cout << "name = " << p_name << " , age = " << p_age << " . " << endl;}
};
}void test0327b() {//Person p("docker", 99);//p.print();
}void test0327c() {Person <string>p("Coco", 33);//体现2,可以对类型赋默认值,使得string数据类型是参数1,默认数据类型是参数2p.print();
}int main() {test0327c();system("pause");return 0;
}

运行结果:

image-20240327200302791

类模板中成员函数创建时机

类模板成员函数和普通类中成员函数创建时机有区别。

  • 普通类成员函数一开始创建。
  • 类模板成员函数调用时创建。

问题:什么叫调用时才创建?

因为具体的类有明确的数据类型,而类模板没有指定的数据类型,所以一开始时不会创建,在实际调用时才会创建。

代码示例:

#include<iostream>
#include<string>
using namespace std;class Per {
public:void print() {cout << "Per类调用print函数" << endl;}
};class Son {
public:void show() {cout << "Son类调用show函数" << endl;}
};template<class T>
class Person {
public:T obj;//这一点在自己编写时忘记了
public:void p_print(){obj.print();}void p_show() {obj.show();}
};void test0327d() {Person <Per> per;per.p_print();//per.p_show();Person <Son> son;//son.p_print();son.p_show();
}
int main() {test0327d();system("pause");return 0;
}

如果不注释per.p_print();son.p_print();,就会报错:image-20240327202523442

注释掉之后,能有运行结果:

image-20240327202555263

类模板对象做函数参数

掌握:类模板实例化出的对象,向函数传参的方式

什么意思?就是将对象作为参数传入函数中。但此时的对象是类模板对象,所以又有模板数据类型不确定的问题在里面。就会有对应三种类模板对象做函数参数的方法。

三种传入方式:

①指定传入类型——直接显示对象的数据类型。

②参数模板化——将对象中的参数变为模板进行传递。

③整个类模板化——将这个对象类型模板化进行传递。

第一种方法,指定传入类型,就是在将类模板对象作为参数传入普通函数时,直截了当的在函数定义时就说明类模板对象的模板数据类型是什么。

第二种,参数模板化——将对象中的参数作为模板传递。

此时函数不再是普通函数,而是模板函数。类模板对象中的模板数据类型需要与模板函数中的模板数据类型保持一致。

第三种,整个类模板化——将这个对象类型作为模板进行传递。

示例代码:

#include<iostream>
#include<string>
using namespace std;template<class T1 , class T2>
class Person {
public:Person(T1 name, T2 age):p_name(name),p_age(age) {}public:T1 p_name;T2 p_age;public:void print() {cout << "name = " << p_name << " , age = " << p_age << " . " << endl;}
};//方式1:指定类模板对象模板数据类型,把类模板对象作为参数传入函数中。
void CommonPrintPerson(Person<string , int> &p) {//现在这是个普通函数p.print();
}//方式2:参数模板化,使用模板函数,把类模板对象中的成员属性类型作为模板数据类型传入。
template<typename T1 , typename T2>
void TemplatePrintPerson(Person <T1, T2>& p) {p.print();
}//方式3:对象模板化,将类模板对象也作为模板数据类型传入函数中
template <class T3>
void TemplateClassPrintPerson(T3 p) {p.print();
}void test0327e() {Person<string, int>p("Jack", 22);Person<string, int>e("Eoa", 32);Person<string, int>r("Ross", 32);//普通函数调用时,指定模板数据类型将类模板对象作为参数传入函数中。CommonPrintPerson(p);TemplatePrintPerson<string, int>(e);TemplateClassPrintPerson(r);
}int main() {test0327e();system("pause");return 0;
}

运行结果:

image-20240327210559275

打印类别

补充的一点是可以通过typeid(要查看的数据类型).name()获得是什么类别。

比如

类模板与继承

当类模板碰到继承时,需要注意以下几点:

如果子类继承的父类是一个类模板时,子类在声明时,需要指定父类中T的类型。如果不指定,编译器无法给子类分配内存。

如果想灵活指定父类中T的类型,子类也需要变为类模板。

为什么不指定父类T中的类型,编译器无法给子类分配内存呢?因为不确定父类有什么数据类型。

所以,对于第一点显式指定父类数据类型,指定的模板数据类型需要写在父类的右边。

image-20240327213010633

如果想要灵活指定父类中的模板数据类型,子类也需要变为类模板。

代码示例如下:

#include<iostream>
#include<string>
using namespace std;template <class T>
class Base {
public:T member;
};//显式指定父类模板数据类型
class Under :public Base<int> {
public:Under() {cout << "父类继承子类,显式指定类模板数据类型" << endl;cout << "类模板数据类型为:" << typeid(member).name() << "."<<endl;}};//灵活指定父类模板,需要子类也是类模板
template <class T1 , class T2>
class Bottom :public Base<T2> {
public:Bottom() {cout << "灵活指定父类模板,需要子类也是类模板" << endl;//cout << "父类类模板数据类型为:" << typeid(member).name() << "." << endl;//需要注释掉才能运行,说明子类先构造,父类才构造。而显式指定的已经指定了。cout << "子类类模板数据类型为:" << typeid(B_member).name() << "." << endl;}
public:T2 B_member;
};void test0327f() {Under u;Bottom<int, char>bo;
}int main() {test0327f();system("pause");return 0;
}

运行结果:

image-20240327213707604

类模板成员函数类外实现

学习目标:掌握类模板外的成员函数类外实现

成员函数类外实现,需要加上类模板的说明,同时加上类模板参数。

代码说明:

#include<iostream>
#include<string>
using namespace std;template<class T1 , class T2 >
class Person {
public:Person(T1 name , T2 age);
public:T1 p_name;T2 p_age;
public:void show();
};template<class T1 , class T2>
Person<T1, T2>::Person(T1 name, T2 age) {this->p_name = name;this->p_age = age;
}template<class T1 , class T2>
void Person<T1 , T2>::show() {cout << "name = " << this->p_name << " , age = " << this->p_age << " . " << endl;
}void test0327g() {Person <string, int>p("Ross", 33);p.show();
}int main() {test0327g();system("pause");return 0;
}

运行结果:

image-20240327221140056

类模板分文件编写

问题:类模板中成员函数创建时机是在调用阶段,导致分文件编写时链接不到。

解决:

①直接包含cpp源文件。

②将声明和实现写入同一个文件中,并改后缀名为.hpp,约定名称,不强制。

类模板中的成员函数创建时机是在调用阶段,调用才会创建,一开始不创建。所以要么包含源码—cpp文件,要么写入同一个文件中,后缀名是’hpp’。

防止头文件重复包含

类模板与友元

学习目标:掌握类模板配合友元函数的类内和类外实现

全局函数类内实现👉直接在类内声明友元即可。

全局函数类外实现👉需要提前让编译器知道全局函数的存在。

我的理解:加了友元标识符之后,类内的函数就变成了全局函数。所以可以在类外直接调用。

弹幕:printPerson函数不加friend时是一个私有的成员函数,但加了friend之后就变成了一个全局函数,因为自身的成员函数默认就可以调用自身的成员属性不需要友元,反之需要友元的都不是成员函数。

全局函数类内实现就一步到位。

代码示例:

#include<iostream>
#include<string>
using namespace std;template<class T1 , class T2>
class Person {//类内全局函数friend void printPerson(Person<T1,T2> p) {cout << "全局函数类内实现" << endl;cout << "name = " << p.p_name << " , age = " << p.p_age << " . " << endl;}
public:Person(T1 name, T2 age) :p_name(name), p_age(age) {}
private:T1 p_name;T2 p_age;
};void test0327h() {Person <string, int> p("张三", 34);printPerson(p);
}int main() {test0327h();system("pause");return 0;
}

运行结果:

image-20240327231909161

全局函数类外实现,由于模板函数的特殊,若将类外模板函数的实现写于声明后,编译器不认识。并且,类内的函数声明是普通的函数声明,需要将其转换成模板函数的声明,

image-20240327232647811

如图,是全局函数类外实现写在了声明后,编译器不认识。

如果写在声明前,由于传入了Person类,编译器也不认识Person类,还需要把Person类的声明也写在前面。

image-20240327232355135

接下来,还是不行,因为这样相当于全局函数是普通函数,需要加上空模板参数列表让编译器认为它是函数模板。

最后,终于成功生成。

image-20240327232714203

完整代码:

#include<iostream>
#include<string>
using namespace std;template<class T1, class T2>
class Person;template<class T1, class T2>
void showPerson(Person<T1, T2> p) {cout << "全局函数类外实现" << endl;cout << "name = " << p.p_name << " , age = " << p.p_age << " . " << endl;
}template<class T1 , class T2>
class Person {//类内全局函数friend void printPerson(Person<T1,T2> p) {cout << "全局函数类内实现" << endl;cout << "name = " << p.p_name << " , age = " << p.p_age << " . " << endl;}friend void showPerson<>(Person <T1, T2> p);
public:Person(T1 name, T2 age) :p_name(name), p_age(age) {}
private:T1 p_name;T2 p_age;
};void test0327h() {Person <string, int> p("张三", 34);//printPerson(p);showPerson(p);
}int main() {test0327h();system("pause");return 0;
}

运行结果:

image-20240327232732006

看弹幕:友元不是类的成员,也不接受它所在区域访问控制级别的约束。

所以,一般都是全局函数类内实现。当然,也可以直接在类里面写上函数模板。

template<class T1 , class T2>
friend void showPerson(Person <T1, T2> p);

一样可以运行。

类模板案例-通用数组实现

制作一个通用的数组类,可以满足如下要求:

  1. 对内置数据类型以及自定义数据类型的数据进行存储。
  2. 将数组中的数据存储到堆区。
  3. 构造函数中可以传入数组的容量。
  4. 提供对应的拷贝构造函数以及operator=防止浅拷贝问题。
  5. 提供尾插法和尾删法对数组中的数据进行增加和删除。
  6. 可以通过下标的方式访问数组中的元素。
  7. 可以获取当前数组中的元素个数和数组的容量。

自己的分析:

①,可以用类模板存储元素。

②,在每个数据存储时,也就是构造函数,需要new一个数据到堆区。同时析构函数需要释放内存。

③,怎么理解?——传入数组容量,才能确定开辟多少内存空间。

④,这个拷贝构造函数,说的是类的拷贝。重写=,就是在类中重写=号运算符。

⑤,尾插法,尾删法。提供两种方法,参数是什么?对于插入,参数就是新的数据元素;对于删除,可以没有参数。内存空间的扩充怎么做?参考职工管理系统。内存空间的删除,就是直接将最后一个元素内存地址释放,并将数组个数-1。

⑥传入对应的数字,获取其东西。

CommonArray.hpp

#include<iostream>
#include<string>
using namespace std;当前元素个数和数组容量。
*/
template <class T>
class CommonArray;template<class T>
class CommonArray {
public:CommonArray(int num);CommonArray(CommonArray& ca);~CommonArray();
public://这个代表什么意思?为什么要是创建一个T类型的东西?这个东西再指向一个数组?T* array;
private:int ca_num;int ca_size;
public:void tail_insert(const T &element);void tail_delete();int get_array_size();int get_element_number();int get_common_array_cost();CommonArray& operator=(const CommonArray& ca);T& operator[](int index);
};template<class T>
CommonArray<T>::CommonArray(int size) {//cout << "有参构造函数调用" << endl;//容量:数组内一共可以放ca_num个元素。this->ca_size = size;//新建一个T类型的数组指针,并把地址设为null//this->array = NULL;//开辟这么大一个空间的数组。this->array = new T[this->ca_size];this->ca_num = 0;
};template<class T>
CommonArray<T>::CommonArray(CommonArray& ca) {//cout << "拷贝构造函数调用" << endl;this->ca_size = ca.ca_size;this->ca_num = ca.ca_num;this->array = new T[ca.ca_size];for (int i = 0; i < ca.ca_size; i++) {this->array[i] = ca.array[i];}
}//拷贝构造函数和重写=运算符防止编译器提供的两个函数造成浅拷贝现象。
//如果浅拷贝,删除时会导致堆区数据重复释放。template<class T>//返回值左值存在,应该要返回引用。
CommonArray<T>& CommonArray<T>::operator=(const CommonArray<T>& ca) {//cout << "operator=函数调用" << endl;this->ca_size = ca.ca_size;this->ca_num = ca.ca_num;this->array = new T[ca.ca_size];for (int i = 0; i < ca.ca_size; i++) {this->array[i] = ca.array[i];}return *this;
}template<class T>
CommonArray<T>::~CommonArray() {if (this->array) {delete[] this->array;this->array = 0;this->ca_size = 0;this->ca_num = 0;}
}//为了防止T被修改,一般用引用的方式传入,使用const
template<class T>
void CommonArray<T>::tail_insert(const T &element) {if (this->ca_num != this->ca_size) {this->array[this->ca_num] = element;this->ca_num += 1;}else {cout << "数组已满,无法新增。" << endl;system("pause");return;}
}template<class T>
void CommonArray<T>::tail_delete() {this->ca_num--;
}template<class T>
int CommonArray<T>::get_array_size() {return this->ca_size;
}template<class T>
int CommonArray<T>::get_element_number() {return this->ca_num;
}template<class T>
int CommonArray<T>::get_common_array_cost() {int cost;cost = sizeof(T) * this->ca_size;return cost;
}template<class T>
//返回值是T是因为要作为返回值,又因为想让它作为左值存在,比如int i = 0 ,就需要让他放回一个引用。。
T& CommonArray<T>::operator[](int index) {return this->array[index];
}

测试文件.cpp

#include<iostream>
#include<string>
using namespace std;#include"CommonArray.hpp"class Person {
public:Person() {}Person(string name, int age) :p_name(name),p_age(age) {}
public:string p_name;int p_age;
};void Print_Array(CommonArray<int>& common_array) {for (int i =0 ; i< common_array.get_element_number(); i++){cout << common_array.array[i] << " ";}
}void Print_Person_Array(CommonArray<Person>& common_array_persons) {for (int i = 0; i < common_array_persons.get_element_number(); i++) {cout << "数组成员" << i + 1 << ": 姓名:" << common_array_persons.array[i].p_name<<"\t,";cout << "\t年龄:" << common_array_persons.array[i].p_age << " \t. " << endl;}}void test0328() {CommonArray<int> ca(5);for (int i = 0; i < ca.get_array_size(); i++) {ca.tail_insert(i);}cout << "ca原先为:" ;Print_Array(ca);ca.tail_delete();cout << "现在的ca容量大小为:" << ca.get_array_size() << endl;cout << "现在的ca 元素个数为:" << ca.get_element_number() << endl;cout << "ca现在为:";Print_Array(ca);cout << endl;CommonArray<Person> persons(5);Person p1("孙悟空", 999);Person p2("韩信", 19);Person p3("爪云", 39);Person p4("张飞", 29);Person p5("安其拉", 18);persons.tail_insert(p1);persons.tail_insert(p2);persons.tail_insert(p3);persons.tail_insert(p4);persons.tail_insert(p5);Print_Person_Array(persons);}int main() {test0328();system("pause");return 0;
}

image-20240402203442273


以上是我的学习笔记,希望对你有所帮助!
如有不当之处欢迎指出!谢谢!

学吧,学无止境,太深了

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/790303.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Google Chrome 常用设置

Google Chrome 常用设置 References 转至网页顶部 快捷键&#xff1a;Home 转至内容设置 chrome://settings/content 清除浏览数据 历史记录 -> 清除浏览数据 关于 Chrome 设置 -> 关于 Chrome chrome://settings/help References [1] Yongqiang Cheng, https:/…

关于ansible的模块 ③

转载说明&#xff1a;如果您喜欢这篇文章并打算转载它&#xff0c;请私信作者取得授权。感谢您喜爱本文&#xff0c;请文明转载&#xff0c;谢谢。 接《关于Ansible的模块①》和《关于Ansible的模块②》&#xff0c;继续学习ansible的user模块。 user模块可以增、删、改linux远…

回溯算法|491.递增子序列

力扣题目链接 class Solution { private:vector<vector<int>> result;vector<int> path;void backtracking(vector<int>& nums, int startIndex) {if (path.size() > 1) {result.push_back(path);// 注意这里不要加return&#xff0c;要取树上…

人脸识别seetaface6 windows + cmake + vs编译,踩坑指南

遇到问题冷静分析&#xff0c;没有解决不了的问题&#xff0c;只是需要时间。与君共勉 环境准备 要在windows 上编译c 源码&#xff0c;需要准备如下软件。省去了详细的安装过程。 visual studio 2022 (社区免费版链接)mingw64 下载路径 (安装后&#xff0c;记得添加系统路径…

elementui 实现一个固定位置的Pagination(分页)组件

系列文章目录 一、elementui 导航菜单栏和Breadcrumb 面包屑关联 二、elementui 左侧导航菜单栏与main区域联动 三、elementui 中设置图片的高度并支持PC和手机自适应 四、 elementui 实现一个固定位置的Pagination&#xff08;分页&#xff09;组件 文章目录 系列文章目录…

今客CRM客户管理系统 v17.3

简介&#xff1a; 今客CRM客户管理系统主要是为了帮助企业解决在日常工作中遇到的客户管理等难题而开发&#xff0c;通过今客CRM客户管理系统可以对企业事务中的不同功能进行操作&#xff0c;用户通过自定义字段类型可以达到适合不同企业的需求。在今客客户关系管理系统中管理…

【子集回溯】【树枝+树层去重】Leetcode 491. 非递减子序列

【组合回溯】【树枝树层去重】Leetcode 491. 非递减子序列 解法1 ---------------&#x1f388;&#x1f388;题目链接&#x1f388;&#x1f388;------------------- 解法1 【树层去重】&#xff1a;如果在同层元素重复(这里无法排序&#xff0c;所以无法使用之前的flag&a…

HarmonyOS 应用开发之启动远程PageAbility(仅对系统应用开放)

启动远程PageAbility同样通过featureAbility中的startAbility接口实现。 除引入’ohos.ability.featureAbility’外&#xff0c;还需引入’ohos.distributedHardware.deviceManager’&#xff0c;通过DeviceManager&#xff08;该组件提供帐号无关的分布式设备的认证组网能力&…

鸿蒙南向开发实战:【智能窗帘】

样例简介 智能窗帘设备不仅接收数字管家应用下发的指令来控制窗帘开启的时间&#xff0c;而且还可以加入到数字管家的日程管理中。通过日程可以设定窗帘开关的时间段&#xff0c;使其在特定的时间段内&#xff0c;窗帘自动打开或者关闭&#xff1b;通过日程管家还可以实现窗帘…

蓝桥杯备考

目录 P8823 [传智杯 #3 初赛] 期末考试成绩 题目描述 输入格式 输出格式 输入输出样例 说明/提示 代码 P8828 [传智杯 #3 练习赛] 直角三角形 题目描述 输入格式 输出格式 输入输出样例 代码 P8833 [传智杯 #3 决赛] 课程 题目背景 题目描述 输入格式 输出格式…

redis---位图Bitmap和位域 Bitfield

位图是字符串类型的拓展&#xff0c;可以使用一个string类型来模拟一个Bit数组。数组的下标就是偏移量&#xff0c;值只有0和1&#xff0c;也支持一些位运算&#xff0c;比如与或非&#xff0c;异或等等&#xff0c;它们的应用场景非常广泛比如可以用来记录用户的签到情况&…

Qt中继承QCheckBox的类结合QTableWidget实现多选并且每个多选的id都不一样

1.相关描述 继承QCheckBox的类MyCheckBox&#xff0c;利用QTableWidget的setCellWidget方式添加MyCheckBox类的对象 2.相关页面 3.相关代码 mycheckbox.h #ifndef MYCHECKBOX_H #define MYCHECKBOX_H#include <QCheckBox> #include <QObject>class MyCheckBox : pu…

力扣刷题Days31-2.两数相关(js)

1&#xff0c;题目 给你两个 非空 的链表&#xff0c;表示两个非负的整数。它们每位数字都是按照 逆序 的方式存储的&#xff0c;并且每个节点只能存储 一位 数字。 请你将两个数相加&#xff0c;并以相同形式返回一个表示和的链表。 你可以假设除了数字 0 之外&#xff0c;…

Vue3:Pinia简介及环境搭建

一、简介 Pinia是Vue3中的状态管理工具&#xff0c;类似与Vue2中的Vuex框架的作用 二、环境搭建 1、安装 npm install pinia2、配置 main.ts import {createApp} from vue import App from ./App.vue // 第一步&#xff1a;引入pinia import {createPinia} from piniacons…

zabbix源码安装

目录 一.安装php和nginx客户端环境 二.修改php配置 三.修改nginx配置文件 四.下载并编译zabbix 五.创建zabbix需要的用户及组 六.安装编译需要的依赖 七.配置zabbix文件 八.数据库配置 九.配置zabbix 十.web界面部署 十一.遇到无法创建配置文件 十二.登录zabbix 前…

2014最新AI学法减分交管12123小程序源码最新玩法

2014最新AI学法减分交管12123小程序源码最新玩法利用ChatGPT实现拍照搜题 利用ChatGPT实现拍照搜题 学法减分这个项目是几年之前的项目&#xff0c;老朋友都知道&#xff0c;以前我用Python实现了向量检索&#xff0c;也就是当时和大家说到的AI题库&#xff0c;那时候国内还没…

鸿蒙实战开发:【实现应用悬浮窗】

如果你要做的是系统级别的悬浮窗&#xff0c;就需要判断是否具备悬浮窗权限。然而这又不是一个标准的动态权限&#xff0c;你需要兼容各种奇葩机型的悬浮窗权限判断。 fun checkPermission(context: Context): Boolean if (Build.VERSION.SDK_INT < Build.VERSION_CODES.M)…

蓝桥杯速成5-AD/DA模数转换

一、原理图 上图可知该芯片使用的是iic时序&#xff0c;而不是51单片机的xpt2046时序&#xff0c;iic我们都很熟悉了吧 并且大赛还提供了我们iic底层驱动代码 左上角有AIN0-4四个转换输入通道&#xff0c;和AOUT一个输出通道&#xff0c;由控制字节选择 地址字节&#xff1a;0x…

自由职业指南大全

现在的自媒体行业风生水起 不知道大家有没有感觉到 无论是网赚圈子还是自媒体圈子&#xff0c;都不要自己去闷头搞&#xff0c;真的没出路 我本人就是最好的例子 前几年都是一个人&#xff0c;盲目乱转&#xff0c;一事无成 现在学会了知识付费&#xff0c;开始慢慢有了成…

硬件RAID横评(上)

正文共&#xff1a;3857字 50图&#xff0c;预估阅读时间&#xff1a;12 分钟 之前误打误撞测试了软件RAID&#xff08;Windows下软RAID测试&#xff09;&#xff0c;发现性能基本上是线性的&#xff0c;而据说硬件RAID性能比这个高的很。那本文将就硬件RAID展开测试&#xff0…