C++初阶类与对象(一):学习类与对象、访问限定符、封装、this指针

2024-01-17 21:36

本文主要是介绍C++初阶类与对象(一):学习类与对象、访问限定符、封装、this指针,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

入门知识已经梳理完毕了,接下来就进入到面型对象的部分学习了


文章目录

  • 1.面向过程和面向对象初步认识
  • 2.类的引入
  • 3.类的定义
    • 3.1类的结构
    • 3.2类的两种定义方式
      • 3.2.1声明和定义全部放在类体中
      • 3.2.2声明和定义分开
    • 3.3成员变量命名规则的建议
  • 4.类的访问限定符及封装
    • 4.1访问限定符
    • 4.2封装
  • 5.类的作用域
  • 6.类的实例化
  • 7.类对象模型
    • 7.1类对象的存储方式
    • 7.2结构体内存对齐规则
  • 8.this指针
    • 8.1this指针的引出
    • 8.2 this指针的特性
    • 8.3经典问题


1.面向过程和面向对象初步认识

C语言典型的面向过程的,关注的是过程,分析出求解问题的步骤,通过函数调用逐步解决问题

请添加图片描述

C++是典型的基于面向对象的,关注的是对象,将一件事情拆分成不同的对象,靠对象之间的交互完成。

将大象和冰箱看作两个对象,每个对象都有自己的特征和行为

首先,冰箱可能有一个开门和关门的方法,而大象可能有一个“装进冰箱”的方法。在这种情况下,你会调用冰箱的“开门”方法,然后调用大象的“装进冰箱”方法,最后再调用冰箱的“关门”方法


2.类的引入

C语言结构体中只能定义变量 。在C++中,结构体内不仅可以定义变量,也可以定义函数。比如:之前在数据结构初阶中,用C语言方式实现的栈,结构体中只能定义变量;现在以C++方式实现,会发现struct中也可以定义函数

之前c:

struct Stack
{int* a;int top;int capacity;//功能函数只能在外面
};

现在C++:

struct Stack
{int* a;int top;int capacity;void Init(size_t capacity){//}void Push(const DataType& data){// 扩容_array[_size] = data;++_size;}//............
};

而C++里sturct会用class来代替

C++兼容c语言struct的所有用法,struct同时升级成了类。 注意:

  1. 类名就是类型,Stack就是类型,不需要加struct。甚至链表里定义next指针时也不需要
  2. 类里面可以定义函数

3.类的定义

3.1类的结构

class ClassName {
public:// 公有成员函数和变量// 可以被类外部访问private:// 私有成员函数和变量// 只能被类内部成员函数访问protected:// 保护成员函数和变量// 类的继承者可以访问
};

class为定义类的关键字,ClassName为类的名字,{ }中为类的主体,注意类定义结束时后面分号不能省略。类体中内容称为类的成员:类中的变量称为类的属性或成员变量 ; 类中的函数称为类的方法或者成员函数

3.2类的两种定义方式

3.2.1声明和定义全部放在类体中

需注意:成员函数如果在类中定义,编译器可能会将其当成内联函数处理(当然这里也是建议,具体还是要看编译器)

class Stack
{
private:int* a;//这三个变量都是声明,内存没有给他们分配空间,创建对象后才有空间int top;int capacity;
public:void Init(size_t capacity)//声明和定义都在类内{a = nullptr;top = -1;capacity = 0;}//............
};

3.2.2声明和定义分开

在类内只进行声明,定义在类外

一般:类声明放在.h文件中,成员函数定义放在.cpp文件中

注意:成员函数名前需要加类名 : :(感觉跟命名空间一个用法)

class Stack
{
private:int* a;int top;int capacity;
public:void Init(size_t capacity);//类内声明//............
};
void Stack::Init(size_t capacity)//类外定义   注意加上  类名::
{a = nullptr;top = -1;capacity = 0;
}

3.3成员变量命名规则的建议

有时候会有这种情况:

class Date
{
private:int year;int mouth;int day;
public:void Init(int year, int mouth, int dat){year = year;//我们大多时会习惯于相同变量名,现在就遇到疑惑了//......}
};

为了避免上述情况,我们习惯于在成员变量前加_(大家默许的规定吧)

int _year;

int _mouth;


4.类的访问限定符及封装

在上面的代码里大家看到了private、public之类的,现在就来讲它们:

4.1访问限定符

C++实现封装的方式:用类将对象的属性与方法结合在一块,让对象更加完善,通过访问权限选择性的将其接口提供给外部的用户使用

请添加图片描述

访问限定符说明:

  1. public修饰的成员在类外可以直接被访问
  2. protected和private修饰的成员在类外不能直接被访问(此处protected和private是类似的)
  3. 访问权限作用域从该访问限定符出现的位置开始直到下一个访问限定符出现时为止
  4. 如果后面没有访问限定符,作用域就到 } 即类结束。
  5. class的默认访问权限为private,struct为public(因为struct要兼容C,C在struct外都能访问)

注意:访问限定符只在编译时有用,当数据映射到内存后,没有任何访问限定符上的区别

4.2封装

我们大家都知道面向对象的三大性质:封装、继承、多态

在类和对象阶段,主要是研究类的封装特性,那什么是封装呢?

封装:将数据和操作数据的方法进行有机结合,隐藏对象的属性和实现细节,仅对外公开接口来和对象进行交互

封装本质上是一种管理,让用户更方便使用类


5.类的作用域

类定义了一个新的作用域,类的所有成员都在类的作用域中。在类体外定义成员时,需要使用 :: 作用域操作符指明成员属于哪个类域

class Date
{
private:int _year;int _mouth;int _day;
public:void Init(int year, int mouth, int day);
};void Date::Init(int year, int mouth, int day)
{_year = year;
}

6.类的实例化

用类类型创建对象的过程,称为类的实例化(类 和 对象是 1对多的关系)

  1. 类是对对象进行描述的,是一个模型一样的东西,限定了类有哪些成员,定义出一个类并没有分配实际的内存空间来存储它;比如:入学时填写的学生信息表,表格就可以看成是一个类,来描述具体学生信息
  2. **一个类可以实例化出多个对象,实例化出的对象占用实际的物理空间,存储类成员变量 **
  3. 类和对象就像是房子和蓝图的关系

请添加图片描述

蓝图没有空间,你怎么能放数据进去呢?

所以需要实例化后,再进行赋值等一系列操作

请添加图片描述


7.类对象模型

7.1类对象的存储方式

只保存成员变量,成员函数存放在公共的代码段(成员函数的地址不在对象中,成员变量是在的)

请添加图片描述

请添加图片描述

结论:一个类的大小,实际就是该类中”成员变量”之和,当然要注意内存对齐

注意空类的大小,空类比较特殊,编译器给了空类一个字节来唯一标识这个类的对象

7.2结构体内存对齐规则

  1. 第一个成员在与结构体偏移量为0的地址处。
  2. 其他成员变量要对齐到某个数字(对齐数)的整数倍的地址处。
    注意:对齐数 = 编译器默认的一个对齐数与该成员大小的较小值 (VS中默认的对齐数为8)
  3. 结构体总大小为:最大对齐数(所有变量类型最大者与默认对齐参数取最小)的整数倍
  4. 如果嵌套了结构体的情况,嵌套的结构体对齐到自己的最大对齐数的整数倍处,结构体的整体大小就是所有最大对齐数(含嵌套结构体的对齐数)的整数倍

关于内存对齐,我这篇文章详细讲了:c语言进阶部分详解(详细解析自定义类型——结构体,内存对齐,位段)


8.this指针

8.1this指针的引出

class Date
{
public:void InitDate(int year, int month, int day){_year = year;_month = month;_day = day;}void PrintDate(){cout << _year << "-" << _month << "-" << _day << endl;}private:int _year;int _month;int _day;
};int main()
{Date d1;Date d2;d1.InitDate(2024, 1, 2);d2.InitDate(2023, 1, 2);d1.PrintDate();d2.PrintDate();return 0;
}

有一个问题:Date类中有 Init 与 Print 两个成员函数,函数体中没有关于不同对象的区分,那当d1调用 Init 函数时,该函数是如何知道应该设置d1对象,而不是设置d2对象

C++中通过引入this指针解决该问题,即:C++编译器给每个“非静态的成员函数“增加了一个隐藏的指针参数,让该指针指向当前对象(函数运行时调用该函数的对象),在函数体中所有“成员变量”的操作,都是通过该指针去访问。只不过所有的操作对用户是透明的,即用户不需要来传递,编译器自动完成

所以实际上:

请添加图片描述

8.2 this指针的特性

  1. this指针的类型:==类的类型 const==,即成员函数中,不能给this指针赋值。*
  2. 只能在“成员函数”的内部使用
  3. this指针本质上是“成员函数”的形参,当对象调用成员函数时,将对象地址作为实参传递给
    this形参。所以对象中不存储this指针
  4. this指针是“成员函数”第一个隐含的指针形参,一般情况由编译器通过ecx寄存器自动传
    递,不需要用户传递
class Date
{
public:// 不能显示的写实参和形参// void Print(Date* const this),这样不行void Print(){//this = nullptr;  this实际上穿过来了cout << this << endl;// 但是可以在类里面显示的使用cout << this->_year << "-" << this->_month << "-" << this->_day << endl;//写出来也没事cout << _year << "-" << _month << "-" << _day << endl;//正常是这样,不用写}private:int _year=1;     // 年int _month=1;    // 月int _day=1;      // 日
};

8.3经典问题

class A
{
public:void print(){cout << "print()";}
private:int _a;
};int main()
{A* a = nullptr;//现在是空指针a->print();return 0;
}

可以运行成功:实际上没有对a进行解引用操作和指向空间的访问

class A
{
public:void print(){cout << "print()" << endl;cout << _a;//多了这个,对a指向的空间访问}
private:int _a;
};int main()
{A* a = nullptr;//现在是空指针a->print();return 0;
}

请添加图片描述


今天就到这里了,下次给大家详细介绍构造函数和析构函数,感谢支持!!!

这篇关于C++初阶类与对象(一):学习类与对象、访问限定符、封装、this指针的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/617287

相关文章

javaSE类和对象进阶用法举例详解

《javaSE类和对象进阶用法举例详解》JavaSE的面向对象编程是软件开发中的基石,它通过类和对象的概念,实现了代码的模块化、可复用性和灵活性,:本文主要介绍javaSE类和对象进阶用法的相关资... 目录前言一、封装1.访问限定符2.包2.1包的概念2.2导入包2.3自定义包2.4常见的包二、stati

C++11范围for初始化列表auto decltype详解

《C++11范围for初始化列表autodecltype详解》C++11引入auto类型推导、decltype类型推断、统一列表初始化、范围for循环及智能指针,提升代码简洁性、类型安全与资源管理效... 目录C++11新特性1. 自动类型推导auto1.1 基本语法2. decltype3. 列表初始化3

C++11右值引用与Lambda表达式的使用

《C++11右值引用与Lambda表达式的使用》C++11引入右值引用,实现移动语义提升性能,支持资源转移与完美转发;同时引入Lambda表达式,简化匿名函数定义,通过捕获列表和参数列表灵活处理变量... 目录C++11新特性右值引用和移动语义左值 / 右值常见的左值和右值移动语义移动构造函数移动复制运算符

C++中detach的作用、使用场景及注意事项

《C++中detach的作用、使用场景及注意事项》关于C++中的detach,它主要涉及多线程编程中的线程管理,理解detach的作用、使用场景以及注意事项,对于写出高效、安全的多线程程序至关重要,下... 目录一、什么是join()?它的作用是什么?类比一下:二、join()的作用总结三、join()怎么

WinForm跨线程访问UI及UI卡死的解决方案

《WinForm跨线程访问UI及UI卡死的解决方案》在WinForm开发过程中,跨线程访问UI控件和界面卡死是常见的技术难题,由于Windows窗体应用程序的UI控件默认只能在主线程(UI线程)上操作... 目录前言正文案例1:直接线程操作(无UI访问)案例2:BeginInvoke访问UI(错误用法)案例

C++中全局变量和局部变量的区别

《C++中全局变量和局部变量的区别》本文主要介绍了C++中全局变量和局部变量的区别,全局变量和局部变量在作用域和生命周期上有显著的区别,下面就来介绍一下,感兴趣的可以了解一下... 目录一、全局变量定义生命周期存储位置代码示例输出二、局部变量定义生命周期存储位置代码示例输出三、全局变量和局部变量的区别作用域

C++中assign函数的使用

《C++中assign函数的使用》在C++标准模板库中,std::list等容器都提供了assign成员函数,它比操作符更灵活,支持多种初始化方式,下面就来介绍一下assign的用法,具有一定的参考价... 目录​1.assign的基本功能​​语法​2. 具体用法示例​​​(1) 填充n个相同值​​(2)

c++ 类成员变量默认初始值的实现

《c++类成员变量默认初始值的实现》本文主要介绍了c++类成员变量默认初始值,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧... 目录C++类成员变量初始化c++类的变量的初始化在C++中,如果使用类成员变量时未给定其初始值,那么它将被

C++中NULL与nullptr的区别小结

《C++中NULL与nullptr的区别小结》本文介绍了C++编程中NULL与nullptr的区别,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编... 目录C++98空值——NULLC++11空值——nullptr区别对比示例 C++98空值——NUL

C++ Log4cpp跨平台日志库的使用小结

《C++Log4cpp跨平台日志库的使用小结》Log4cpp是c++类库,本文详细介绍了C++日志库log4cpp的使用方法,及设置日志输出格式和优先级,具有一定的参考价值,感兴趣的可以了解一下... 目录一、介绍1. log4cpp的日志方式2.设置日志输出的格式3. 设置日志的输出优先级二、Window