C++ Day3
- C++对C的结构体的扩充
- 类
- this指针
- 类的大小
- 类中的特殊成员函数
- 构造函数
- 作业
C++对C的结构体的扩充
- C语言中的结构体,仅仅只是属性(变量)的聚合体,不可以在结构体中定义行为(函数)。如果非要在结构体中定义行为,则需要定义函数指针,并使用回调函数来实现。
- C++对C语言中的结构体,进行了翻天覆地的变化
- 在C++中的结构体中可以定义函数,而C语言中不可以,C++中结构体内定义函数往往是,结构体内声明,结构体外定义
- C++中的结构体,在定义变量时,可以不用加struct,而C语言中的结构体必须要加上
- C++中的结构体中可以再定义一个新的结构体类型,并且可以在外界直接使用,而C语言中的结构体不可以
- C++中的结构体,有访问权限,分别是public、private、protected,默认是public权限,C语言中的结构体没有访问权限
- C++中的结构体,可以被继承,继承方式有public、private、protected,默认是public权限,C语言中的结构体,不能被继承
- C++中的结构体,在声明类型时,可以给成员属性(变量)初始值,而C语言中的结构体不允许
- C++中的结构体,会默认提供一些特殊成员函数,如析构函数、拷贝赋值函数、移动赋值函数。C中的结构体没有
类
- 类的访问权限有三种:public、protected、private
类的默认权限为private - 类中可以有成员属性和成员函数,一般成员属性设置成私有、成员变量设置为公共的
- 类的成员函数可以访问类中所有权限下的成员,包括私有成员
- 每种访问权限可以在类中出现多次,从关键字开始一直到下一个关键字出现或者类体结束
- 如果非要在类外对私有或受保护的成员进行读取或写入,需要在类内提供相关的公共接口来操作
- 类的访问权限是针对于类体而言,而不是针对于类对象而言
定义格式
//一般出现在头文件中
class 类名 {成员类型 成员名;...成员函数声明;//例如void show();
};//一般出现在源文件中
void 类名::show()
{函数体;
}
this指针
- this指针是类中系统为所有非静态成员函数提供的一个隐藏的形参指针,指代当前对象的起始地址
- 指代当前对象,哪个对象使用我,我就指向哪个对象
- 在非静态成员函数中,如果调用了类中的成员(成员函数、成员变量),即使没有加this,系统也默认加了this。
- this的原型: 类名 * const this;
- 必须使用this的情况:
- 当形参名和成员名同名时,可以使用this用于区分
- 在拷贝赋值函数中,用于返回自身引用时,必须使用this
类的大小
- 一个空类的大小为 1 字节,用于占位作用,后期如果有成员变量,就会将该1字节空间分配给成员使用
- 类中的成员函数不占类体的大小,成员函数只有被调用时,才会分配空间
- 只有类的成员属性才占内存空间,其类的大小要遵循字节对齐原则
- 如果类中有虚函数,则会多一个虚指针的空间
类中的特殊成员函数
- 当定义一个空类时,即使没有手动定义某些函数,C++系统会默认自动为其提供一些特殊成员函数
- 这些函数也可以程序员手动定义,当手动定义后,系统就不再提供默认的了
- 这些函数也无需程序员手动调用,在特殊时期,C++系统会自动调用
- 种类:构造函数、析构函数、拷贝构造函数、拷贝赋值函数、移动构造函数、移动赋值函数、取地址运算符重载函数
构造函数
- 构造函数是在使用类实例化对象时,用于给类对象分配内存空间以及对成员进行初始化工作时使用
- 调用时机:无需程序员手动调用,当使用一个类实例化对象时,系统自动调用构造函数
- 调用格式:
无参构造: 类名 对象名;
例如:string s1;
有参构造: 类名 对象名(实参列表);
例如:string s2(“hello a”);
string s3(5, ‘A’); - 一个类中可以定义多个构造函数,要求参数必须不同,这些构造函数构成重载关系,虽然一个类中可以定义多个构造函数,但是一个类对象的只能使用一个构造函数构造出来
- 一个类中,如果没有显性定义任何构造函数,那么系统会默认提供一个无参构造函数,但凡类中提供任意一个构造函数,系统就不再提供默认的无参构造函数了,如果非要使用无参构造函数,需要程序员手动定义无参构造函数
- 构造函数的形参列表也可以设置默认参数,但是要注意是否跟其他重载的函数冲突
- 构造函数的初始化工作是在初始化列表中完成的。
初始化列表:在构造函数函数头部后面,由冒号引出,完成初始化工作
格式: 类名(形参类型1 形参1, 形参类型2 形参 - 必须使用构造函数的初始化列表的情况
- 如果类的成员变量中有常成员变量,对该成员的初始化工作,必须使用初始化列表完成
- 如果类中有引用成员,那么对该成员也必须使用初始化列表完成
- 如果构造函数的形参名和成员变量同名时,可以使用初始化列表来解决
- 如果一个类中,有另一个类的对象,对该成员对象的初始化工作需要在初始化列表中显性定义该成员对象的有参构造,否则系统会自动调用其无参构造
- 定义格式
class stu {
public: //类名(形式参数类别): 初始化列表 {函数体;}stu(string name, int age, double score) : name(name) , age(age) , score(socre) {函数体;}
};
作业
使用C++手动封装一个顺序表
//main.cpp
#include <iostream>
#include "seqlist.h"using namespace std;int main()
{//创建一个长度为5的顺序表Seqlist A(5);//循环输入30个数据,顺序表满时将会自动扩容for(int i = 0; i < 30; i++){A.insert_tail(i + 1);}cout << "初始顺序表" << endl;A.show();cout << "顺序表长度为:" << A.getlen() << endl;cout << "将22插入到第3位后" << endl;A.insert_pos(22,3);A.show();cout << "顺序表长度为:" << A.getlen() << endl;cout << "删除第二个元素后" << endl;A.delete_pos(2);A.show();cout << "顺序表长度为:" << A.getlen() << endl;cout << "删除最后一个元素后" << endl;A.delete_tail();A.show();cout << "第5个元素是:" << A.at(5) << endl;cout << "顺序表长度为:" << A.getlen() << endl;return 0;
}
//seqlist.h
#ifndef SEQLIST_H
#define SEQLIST_H#include <iostream>using namespace std;using datatype = int;class Seqlist
{
private:datatype *data;int size = 0;int len = 0;
public://无参构造,默认创建长度为10的顺序表Seqlist() :data(new datatype[10]), size(10), len(0){}//有参构造,创建指定长度的顺序表Seqlist(int size) :data(new datatype[size]), size(size), len(0){}//判空bool is_empty();//判满bool is_full();//尾插bool insert_tail(datatype e);//指定位置插入元素bool insert_pos(datatype e, int pos);//获取长度int getlen();//尾删bool delete_tail();//指定位置删除元素bool delete_pos(int pos);//访问任意一个元素datatype &at(int pos);//扩容函数void expend();//展示函数void show();//释放顺序表void free();
};#endif // SEQLIST_H
//seqlist.cpp
#include "seqlist.h"bool Seqlist::is_empty(){//判断实际长度为0即是表空if(len == 0){return true;}else{return false;}
}bool Seqlist::is_full(){//判断实际长度为最大值size时即是表满if(len == size){return true;}else {return false;}
}bool Seqlist::insert_tail(datatype e)
{//表满时调用expend扩容顺序表if(is_full()){expend();}//将数据存储在下标为实际长度len的元素中data[len] = e;len++;return true;
}bool Seqlist::insert_pos(datatype e, int pos)
{//表满时扩容if(is_full()){expend();}//插入必须在表的已用范围内if(pos < 0 || pos > len){cout << "位置非法" << endl;return false;}else{//从表尾开始将每个元素循环后移,给新数据腾出位置for(int i = len - 1;i >= pos - 1; i--){data[i+1] = data[i];}//插入新数据data[pos-1] = e;len++;return true;}
}//直接返回len
int Seqlist::getlen()
{return len;
}
//尾删
bool Seqlist::delete_tail()
{//表空时失败if (is_empty()) {return false;}len--;return true;
}//按位置删除
bool Seqlist::delete_pos(int pos)
{//表空时失败if(is_empty()){cout << "表已空" << endl;return false;}else if(pos < 0 || pos >= size){ //位置违法时失败cout << "位置非法" << endl;return false;}else{//从要删除的位置循环前移来删除for (int i = pos - 1; i < len; i++) {data[i] = data[i+1];}len--;return true;}
}//访问指定位置的函数
datatype &Seqlist::at(int pos)
{//位置非法时失败if(pos < 0 || pos >= size){return data[-1];}else{return data[pos-1];}
}//二倍扩容函数
void Seqlist::expend()
{//申请之前2倍的空间datatype *temp = new datatype[2*size];//将之前的数据拷贝到新空间memmove(temp, data, sizeof (datatype)*size);//释放之前的空间delete data;//将指针指向新的空间data = temp;//将temp指针置空temp = NULL;//顺序表容量扩容到2倍size *= 2;
}//遍历顺序表
void Seqlist::show()
{//遍历顺序表,每输出5个数据换行一次int count = 0;for (int i = 0;i < len; i++) {cout << data[i] << '\t';count++;if(count%5 == 0){cout << endl;}}if(count%5){cout << endl;}
}void Seqlist::free()
{//释放顺序表空间delete data;//指针置空data = NULL;
}
运行结果