专业做涂料网站,太原网站关键词优化,wordpress 透明,wordpress中view不见了目录
一. 前言
二. 面向对象与面向过程 2.1 面向过程 2.2 面向对象
三. 类的基础知识
3.1 类的引入
3.2 类的定义
3.3 成员变量的命名规则
3.4 封装
3.5 类的访问限定符
3.6 类的作用域
3.7 类的实例化
四. 类的对象模型
4.1 类对象的大小
4.2 类对象的存储方式 … 目录
一. 前言
二. 面向对象与面向过程 2.1 面向过程 2.2 面向对象
三. 类的基础知识
3.1 类的引入
3.2 类的定义
3.3 成员变量的命名规则
3.4 封装
3.5 类的访问限定符
3.6 类的作用域
3.7 类的实例化
四. 类的对象模型
4.1 类对象的大小
4.2 类对象的存储方式
4.3 空类的大小
五. this指针
5.1 this指针的引出
5.2 this指针的特性
5.3 小试牛刀 一. 前言 前几期我们介绍了C相比C语言新增的一些语法相信大家已经对C有了一定的认知。而从本期开始我们将正式进入C类和对象的学习感受C基于面向对象编程的魅力。在学习过程中我们将接触到面向对象的三大特性之一封装。
二. 面向对象与面向过程 在学习编程的过程中各位想必或多或少都听说过这两个概念。都知道C语言是面向过程的C、Jave等语言是面向对象的那么究竟什么是面向过程而面向对象又是什么意思呢 2.1 面向过程 C语言是面向过程的关注的是实现的过程分析出求解问题的步骤通过函数调用逐步解决问题。 就好像我们要洗衣服从面向过程的角度来洗衣服的流程图就像下面所示 又或者我们要设计一个外卖点餐系统从面向过程的角度我们应该设计类似下面的流程 总结面向过程关注的是一个个步骤例如放衣服、手搓以及用户下单等等通过将这些具体的步骤一步步在函数中实现使用时再依次进行调用即可。 2.2 面向对象 而C是基于面向对象的关注的是对象将一件事情拆分成不同的对象靠对象之间的交互完成。 回到洗衣服面向对象关注的就只有四个对象人、衣服、洗衣粉和洗衣机。人只需将衣服和洗衣服放入洗衣机中即可。至于洗衣机是如何洗衣服、是如何甩干的我们无需关心。 而对于外卖点餐系统我们关注的也不是分配骑手、骑手送餐这些具体的步骤而是关注骑手、商家和用户这三个对象之间的交互对用户如何下单、骑手如何送餐并不关心。 总结面向过程关注的完成某件事的对象例如衣服、洗衣机以及骑手等等。通过描叙这些对象在整件事中的关系和行为最终得以解决问题。 三. 类的基础知识
3.1 类的引入 在C中类是用来描述对象的是一种用户自定义的数据类型。在C语言中结构体就是种自定义类型但其只能用来定义变量。而在C中结构体被升级成了类其不仅可以定义成员变量还可以定义成员函数。如下
//实现一个栈类
typedef int DataType;
struct Stack
{void Init(size_t capacity){//栈初始化}void Push(const DataType data){//栈的插入}DataType Top(){//取栈顶元素}void Destroy(){//栈空间销毁}DataType* _array;size_t _capacity;size_t _size;
}; 而在C中我们更喜欢用class关键字来替代struct
typedef int DataType;
class Stack //用class来定义一个类
{//成员函数、类方法void Init(size_t capacity){}void Push(const DataType data){}DataType Top(){}void Destroy(){}//成员变量、类属性DataType* _array;size_t _capacity;size_t _size;
};
3.2 类的定义 类的结构如下所示
class className //class关键字类名
{// 类主体由成员函数和成员变量组成}; // 后面的分号不要漏
class为定义类的关键字ClassName为类的名字{}中为类的主体注意类定义结束时后面分 号不能省略。类主体中内容称为类的成员类中的变量称为类的属性或成员变量; 类中的函数称为类的方法或者成员函数。 类的定义方式有两种声明和定义结合、声明和定义分离。 声明和定义结合 即声明和定义都放在类主体中如下
class Date
{//成员函数的声明定义void Print(){cout _year 年 _month 月 _day 日 endl;}//成员变量的声明int _year;int _month;int _day;
}; 注意成员函数如果在类中定义编译器可能会将其当成内联函数处理。 声明和定义分离 即类声明放在.h文件中成员函数定义放在.cpp文件中。一般我们会更推荐采用这种分文件编程的方式
//class.h文件
class Date
{//成员函数的声明void Print();//成员变量的声明int _year;int _month;int _day;
};//class.cpp文件
void Date::Print()
{cout _year 年 _month 月 _day 日 endl;
} 注意类外定义的成员函数名前需要加类名类作用限定符:: 3.3 成员变量的命名规则 我们先来看看一个别扭的代码
class Date
{void Init(int year){// 这里的year到底是成员变量还是函数形参year year;}int year;
}; 由于Init函数的形参也为year编译器会优先将year认为是函数形参最终相当于将自身的值赋给自身与本意违背。 为了避免上面命名冲突的情况发生我们通常会给成员变量加上前缀或者后缀加以区分。如下所示
class Date
{void Init(int year){_year year;}int _year; //前缀
};
// 或者这样
class Date
{void Init(int year){year_ year;}int year_; //后缀
};// 其他方式也可以的只要可以加以区分即可一般都是加个前缀或者后缀就行。
3.4 封装 面向对象具有三大特性封装、继承、多态。在类和对象中我们主要接触到的就是封装那么究竟什么是封装呢 在类的设计时我们通常不希望使用者直接访问类中的成员变量而是仅通过使用我们在类中设计的接口函数来对对象进行交互。这种隐藏对象的属性和实现细节将数据和操作数据的方法进行有机结合仅对外公开接口来和对象进行交互就称作封装。 封装本质上是一种管理是为了让用户更方便地使用类无需关注复杂的底层实现细节。 举个栗子对于电脑这样一个复杂的设备对于计算机使用者而言不用关心内部核心部件比如主板上线路是如何布局的CPU内部是如何设计的等用户只需要知道怎么开机、怎么通过键盘和鼠标与计算机进行交互即可。因此计算机厂商在出厂时对计算机进行了封装在外部套上壳子将内部实现细节隐藏起来仅仅对外提供开关机、鼠标以及键盘插孔等让用户可以与计算机进行交互即可 3.5 类的访问限定符 在C中实现封装我们可以通过类将数据和操作数据的方法进行有机结合再通过访问权限来隐藏对象内部实现细节并控制哪些方法可以在类外部直接使用。 C可以通过访问限定符来控制访问权限访问限定符有如下三种 public修饰的成员在类外可以直接被访问protected和private修饰的成员在类外不能直接被访问二者的区别要在后面学习继承时才会体现这里可以粗略认为它们是类似的。 具体使用方式如下所示
class Date
{
public: //使用访问限定符加冒号限定变量或函数的访问权限void Print(){cout _year 年 _month 月 _day 日 endl;}int _year 10; //C11支持给成员变量缺省值protected:int _month 10;private:int _day 10;
};int main()
{Date d; //类的实例化类名变量名d._year 2023; //_year是共有的类外可以访问//无法通过编译保护和私有变量不能在类外访问//d._month 10;//d._day 10;d.Print(); //Print函数是共有的类外可以访问
} 注意事项 1、访问限定符的作用域是从该访问限定符出现的位置开始直到下一个访问限定符出现时为止 2、如果后面没有访问限定符作用域就到 } 即类结束处。 3、在类内对成员进行访问不受访问限定符限制 4、 使用class定义的类的默认访问权限为private而使用struct为public(因为struct要兼容CC语言的结构体成员是允许外部访问的) 3.6 类的作用域 类定义了一个新的作用域简称类域类的所有成员都在当前类域中。在类体外定义成员时需要加上::作用域操作符指明成员属于哪个类域如果没有加上作用域操作符则编译器默认只会在全局进行定义。
class Person
{
public:void PrintPerson();
private:char _name[20];char _gender[3];int _age;
};//这里定义的PrintPerson()是全局函数
void PrintPerson()
{cout void PrintPersonInfo() endl;
}//这里定义的PrintPerson()是Person类中的成员函数
void Person::PrintPersonInfo()
{cout void Person::PrintPerson() endl;
}int main()
{Person p;PrintPerson(); //调用全局的p.PrintPerson(); //调用类域中的
} 3.7 类的实例化 用类类型来创建对象的过程称作类的实例化。类是对对象进行描述的是一个像模型一样的东西限定了类有哪些成员定义一个类并没有分配实际的内存空间来存储它类中的成员变量仅仅只是声明。 一个类可以实例化出多个对象实例化出的对象会占用实际的内存空间用来存储类中的成员变量。举例如下
class Person //Person类的定义
{
public:void PrintPerson(){cout _name _gender _age endl;}char _name[20]; //这里的成员变量都是声明char _gender[3];int _age;
};
int main()
{Person p; //实例化一个对象pp._age 20; //p是类实例化出来的对象占用内存空间顾可以对成员变量_age进行操作//下面的写法均错误类中的_age只是声明没有内存空间Person::_age 20;Person._age 20;return 0;
} 做个比方类就好比一张建筑设计图类实例化对象就好比现实中使用建筑设计图建造房子每栋房子就相当于一个对象一张建筑设计图可以建造出许多栋房子。建筑图纸本身没有空间无法住人只有用建筑图纸建造出来的房子才具有空间用来住人。同样类也只是设计实例化出的对象才能实际存储数据占用内存空间。 四. 类的对象模型
4.1 类对象的大小 一个类中既可以有成员变量也可以有成员函数那么一个类实例化出来的对象中究竟包含了什么我们可以用sizeof操作符来计算一个类对象的大小
class A
{
public:void PrintA(){cout _a endl;}
private:char _a;
};int main()
{A a;cout sizeof(a) endl;return 0;
} 我们看到最终结果为1为什么呢这就要谈到类对象在内存中的存储方式了。 4.2 类对象的存储方式 一种最简单的方式就是将成员变量和成员函数全部包含在对象中但是这也会引来一个问题
int main()
{A a;A b;A c;a.PrintA();b.PrintA();c.PrintA();return 0;
} 当我们实例化出多个对象时每个对象中的成员变量是不同的但调用的是同一个函数。如果按照此种方式存储当一个类创建多个对象时每个对象中都会保存一份函数代码相同代码保存多次浪费空间。 我们在4.1的例子可以看出C并不是以这种方式来存储的很明显PrintA()并没有存储在类对象中类对象中只有一个大小为1字节的成员变量_a。 我们说类就像建筑设计图类对象就像一栋栋房子。类中的成员变量可以看做居民居民需要居住在房子中占据房子空间每栋房子里的居民不同而成员函数就像小区中的娱乐设施如游泳池、篮球场等等它们是小区中所有住户的公共资源只有一份相互共享。 娱乐设施建造在小区之中而我们的成员函数保存的地方就是内存中的公共代码区所有对象共享这一份代码大大节省了内存空间。存储方式如下图所示 结论一个类的大小实际就是该类中”成员变量”之和与成员函数无关。而成员变量的存储方式和结构体一样需要遵循内存对齐。 有关内存对齐的知识可以参考往期文章【C语言】你真的了解结构体吗http://t.csdn.cn/sqzTO 4.3 空类的大小 我们先来看看如下代码
// 类中既有成员变量又有成员函数
class A1 {
public:void f1() {}
private:int _a;
};// 类中仅有成员函数
class A2
{
public:void f2() {}
};
// 类中什么都没有---空类
class A3
{};int main()
{cout sizeof(A1) endl; //既有成员变量又有成员函数cout sizeof(A2) endl; //仅有成员函数cout sizeof(A3) endl; //什么也没有return 0;
} A1有一个int类型的成员变量占四个字节这毫无疑问。但我们发现A2和A3尽管它们没有成员变量它们却也占了1个字节的存储空间这和上面说的结论不一样呀这一个字节的空间到底从何而来难道是成员函数不不不这实际上是编译器对空类的特殊处理 特殊处理空类也可以实例化出对象为了标识对象的存在编译器会给这个空对象分配一个字节的存储空间用于占位。故空类实例化出的对象大小为1个字节。 五. this指针
5.1 this指针的引出 上面我们说过类中的成员函数保存在公共代码区中那么当一个对象调用成员函数成员函数又是如何识别对象并进行操作呢下面我们先来定义一个日期类
class Date
{
public:void Init(int year, int month, int day){_year year;_month month;_day day;}void Print(){cout _year - _month - _day endl;}
private:int _year; // 年int _month; // 月int _day; // 日
};
int main()
{Date d1, d2;d1.Init(2023, 8, 21);d2.Init(2024, 8, 21);d1.Print();d2.Print();return 0;
} 上面的代码中Date类有 Init 与 Print 两个成员函数但是函数体中没有关于不同对象的区分那当d1调用 Init 函数时该函数是如何知道应该设置d1对象而不是设置d2对象呢 C中通过引入this指针来解决该问题即C编译器给每个“非静态的成员函数“增加了一个隐藏的指针参数让该指针指向当前对象(函数运行时调用该函数的对象)在函数体中所有对成员变量的操作都是通过该指针去访问。只不过所有的操作对用户是透明的即用户不需要来传递编译器自动完成。 例如上面的 Init 成员函数实际上是如下的形式
//this指针作为隐藏参数指向调用的对象
void Init(Date* const this, int year, int month, int day)
{this-_year year; //通过this指针找到对象对其内容进行修改this-_month month;this-_day day;
}
5.2 this指针的特性
this指针的类型类类型* const即成员函数中不能修改this指针。this指针只能在“成员函数”的内部使用this指针本质上是“成员函数”的形参当对象调用成员函数时将对象的地址作为实参传递给 this形参。所以对象中不存储this指针this指针是“成员函数”第一个隐含的指针形参一般情况由编译器通过ecx寄存器自动传 递不需要用户传递。顾this指针存储在ecx寄存器中 5.3 小试牛刀 学了this指针的特性我们来两道题目来练练手 Q1下面程序编译运行结果是A、编译报错 B、运行崩溃 C、正常运行
class A
{
public:void Print(){cout Print() endl;}
private:int _a;
};
int main()
{A* p nullptr;p-Print();return 0;
} 答案是C程序正常运行。由于Print()是个成员函数存放在公共代码区因此编译器不会到p所指向的对象中去调用函数而是直接调用公共代码区中的函数然后将p作为this指针传入Print()函数。在Print()函数中由于只有一条输出语句故程序可以正常运行。 Q2下面程序编译运行结果是A、编译报错 B、运行崩溃 C、正常运行
class A
{
public:void PrintA(){cout _a endl;}
private:int _a;
};
int main()
{A* p nullptr;p-PrintA();return 0;
} 答案是B程序运行崩溃。与前一个程序不同的是PrintA()函数输出的是成员变量。由于p调用PrintA()函数时传入的this指针为nullptr而访问成员变量_a实际上是通过this-_a来进行访问编译器只是将this进行了隐藏这无疑是一种对空指针的解引用故程序运行时会崩溃。 以上就是本期的全部内容啦
制作不易能否点个赞再走呢