案例:自定义数组类
需求:
1,对内置数据及自定义数据类型的数据存储
2,将数组中的数据存储到堆区
3,构造函数中可以存入数组的容量
4,提供对应的拷贝构造函数和=运算符重载防止浅拷贝问题的发生
5,提供尾插法和尾删法对数组中的数据进行增加和删除
6,提供下标的方式访问数组中的元素
7,获取数组中当前元素个数和数组的容量
代码的注释写的很详细了,个人觉得没啥好解释的,认真看看,都能看懂,很经典的案例,值得学习!!!
一、yy_Array.hpp
yy_Array.hpp
文件包括:自定义数组类以及类的成员函数、构造函数、析构函数等相应实现
//防止头文件重复包含
#pragma once#include<iostream>
template<class T>
class yy_Array
{
public:yy_Array(int capacity)//有参构造函数,传入数组初始的容量{std::cout << "有参构造" << std::endl;this->capacity_ = capacity;this->size_ = 0; //数组刚开始一个元素都没有this->p_address_ = new T[this->capacity_];//开辟一块堆区空间,大小为数组的容量}//拷贝构造(为例防止堆区带来的浅拷贝问题)//若属性有在堆区开辟的,一定要用自己的拷贝构造函数,防止系统默认使用浅拷贝带来的问题//这里需要在堆区开辟数组,故有在堆区开辟的属性,需要使用自己的拷贝构造函数yy_Array(const yy_Array& arr) {std::cout << "拷贝构造" << std::endl;this->capacity_ = arr.capacity_;this->size_ = arr.size_;this->p_address_ = new T[arr.capacity_]; //因为是堆内存,要重新开辟空间,深拷贝//若数组中本身就有数据,在深拷贝时,也需要拷贝过来for (int i = 0; i < this->size_; i++) {this->p_address_[i] = arr.p_address_[i];}}//赋值运算符重载operator=,也是为了防止浅拷贝问题yy_Array& operator=(const yy_Array& arr) //返回本身,因为实际系统的赋值运算符支持a=b=c{std::cout << "运算符重载operator=" << std::endl;//先判断原来堆区中是否有数据,若事前有数据,先释放掉if (this->p_address_ != NULL) {delete[] this->p_address_;//释放这个指向数组的指针this->p_address_ = NULL;//指针置空,防止野指针this->capacity_ = 0;this->size_ = 0;}//深拷贝,从堆区开辟内存this->capacity_ = arr.capacity_;this->size_ = arr.size_;this->p_address_ = new T[arr.capacity_]; //因为是堆内存,要重新开辟空间,深拷贝//若数组中本身就有数据,在深拷贝时,也需要拷贝过来for (int i = 0; i < this->size_; i++){this->p_address_[i] = arr.p_address_[i];}return *this;//返回自身}//尾插法void Push_Back(const T& val) {//若容量已经满了,进行提示if (this->capacity_ == this->size_) {std::cout << "数组已满" << std::endl;return;}else {this->p_address_[this->size_] = val;//传入的val插到数组最后一个this->size_++;//插入一个元素,数组大小要进行更新}}//尾删法void Pop_Back()//删除操作不需要传参,直接把最后一个元素干掉即可,也就是让用户访问不到即可{//要进行尾删法,首先数据中必须有元素if (this->size_ == 0) {std::cout << "数组已为空" << std::endl;return;}else {this->size_--;//用户访问不到最后一个元素即可}}//通过下标的方式进行访问数组中的元素//因为是自己写的模板数组,单纯的a[0]是无法进行访问的,故需要重新对[]运算符进行重载T& operator[](int index) {return this->p_address_[index];}//返回数组的容量int getCapacity() {return this->capacity_;}//返回数据的大小int getSize() {return this->size_;}~yy_Array()//析构函数{if (p_address_ != NULL) {std::cout << "析构函数" << std::endl;delete[] this->p_address_;//释放这个指向数组的指针this->p_address_ = NULL;//指针置空,防止野指针}}private:T* p_address_; //指针指向堆区开辟的真实数组int capacity_;//数组的容量,即数组最多存放多少元素int size_;//数组当前有多少个元素
};
二、array.cpp
主函数入口,分成三个函数(test1()、test2()以及test3())进行不同功能的测试
调试学习时,将主函数中test1()
、test2()
、test3()
挨个取消注释,运行单个测试函数进行学习
#include <iostream>
#include "yy_Array.hpp"void printIntArr(yy_Array<int>& arr) //打印输出数组内的元素
{for (int i = 0; i < arr.getSize(); i++) {std::cout << arr[i] << std::endl;}
}void test1()
{yy_Array<int>yy_arr_1(5);//测试 有参构造函数 和 析构函数yy_Array<int>yy_arr_2(yy_arr_1);//测试 拷贝构造函数 和 析构函数yy_Array<int> yy_arr_3(100);//测试 有参构造 和 析构函数yy_arr_3 = yy_arr_1;//测试 运算符重载operator= 和 析构函数
}void test2()
{yy_Array<int>yy_arr_1(10);//测试 有参构造函数 和 析构函数//测试尾插法,利用尾插法向数组中插入数据for (int i = 0; i < 10; i++) {yy_arr_1.Push_Back(i);}std::cout << "yy_arr_1的元素为:" << std::endl;printIntArr(yy_arr_1);//测试数组容量std::cout << "yy_arr_1的容量为:" << yy_arr_1.getCapacity() << std::endl;//测试数组大小std::cout << "yy_arr_1的目前大小为:" << yy_arr_1.getSize() << std::endl;yy_Array<int>yy_arr_2(yy_arr_1);//测试 拷贝构造函数 和 析构函数std::cout << "yy_arr_2的元素为:" << std::endl;printIntArr(yy_arr_2);//测试数组容量std::cout << "yy_arr_2的容量为:" << yy_arr_2.getCapacity() << std::endl;//测试数组大小std::cout << "yy_arr_2的目前大小为:" << yy_arr_2.getSize() << std::endl;//测试尾删法yy_arr_2.Pop_Back();//测试数组容量std::cout << "yy_arr_2尾删后的容量为:" << yy_arr_2.getCapacity() << std::endl;//测试数组大小std::cout << "yy_arr_2尾删后的目前大小为:" << yy_arr_2.getSize() << std::endl;
}//测试自定义数据类型
class Beyond
{
public:Beyond() {};//无参构造函数Beyond(std::string name, int age){this->name_ = name;this->age_ = age;}std::string name_;int age_;
};
//输出自定义数据类型
void printBeyondArr(yy_Array<Beyond>& arr)
{for (int i = 0; i < arr.getSize(); i++){std::cout << "姓名:" << arr[i].name_ << "年龄:" << arr[i].age_ << std::endl;}
}
void test3()
{yy_Array<Beyond> arr(5);Beyond hjj("黄家驹", 31);Beyond hjq("黄家强", 56);Beyond hgz("黄贯中", 57);Beyond ysr("叶世荣", 58);//将数据通过尾插法插入到数组中arr.Push_Back(hjj);arr.Push_Back(hjq);arr.Push_Back(hgz);arr.Push_Back(ysr);//输出数组数据printBeyondArr(arr);//测试数组容量std::cout << "arr的容量为:" << arr.getCapacity() << std::endl;//测试数组大小std::cout << "arr的目前大小为:" << arr.getSize() << std::endl;
}int main(int argc, char* argv[])
{test1();test2();test3();return 0;
}