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

相关文章

本地搭建DeepSeek-R1、WebUI的完整过程及访问

《本地搭建DeepSeek-R1、WebUI的完整过程及访问》:本文主要介绍本地搭建DeepSeek-R1、WebUI的完整过程及访问的相关资料,DeepSeek-R1是一个开源的人工智能平台,主... 目录背景       搭建准备基础概念搭建过程访问对话测试总结背景       最近几年,人工智能技术

C++实现封装的顺序表的操作与实践

《C++实现封装的顺序表的操作与实践》在程序设计中,顺序表是一种常见的线性数据结构,通常用于存储具有固定顺序的元素,与链表不同,顺序表中的元素是连续存储的,因此访问速度较快,但插入和删除操作的效率可能... 目录一、顺序表的基本概念二、顺序表类的设计1. 顺序表类的成员变量2. 构造函数和析构函数三、顺序表

Ollama整合open-webui的步骤及访问

《Ollama整合open-webui的步骤及访问》:本文主要介绍如何通过源码方式安装OpenWebUI,并详细说明了安装步骤、环境要求以及第一次使用时的账号注册和模型选择过程,需要的朋友可以参考... 目录安装环境要求步骤访问选择PjrIUE模型开始对话总结 安装官方安装地址:https://docs.

使用C++实现单链表的操作与实践

《使用C++实现单链表的操作与实践》在程序设计中,链表是一种常见的数据结构,特别是在动态数据管理、频繁插入和删除元素的场景中,链表相比于数组,具有更高的灵活性和高效性,尤其是在需要频繁修改数据结构的应... 目录一、单链表的基本概念二、单链表类的设计1. 节点的定义2. 链表的类定义三、单链表的操作实现四、

Go语言利用泛型封装常见的Map操作

《Go语言利用泛型封装常见的Map操作》Go语言在1.18版本中引入了泛型,这是Go语言发展的一个重要里程碑,它极大地增强了语言的表达能力和灵活性,本文将通过泛型实现封装常见的Map操作,感... 目录什么是泛型泛型解决了什么问题Go泛型基于泛型的常见Map操作代码合集总结什么是泛型泛型是一种编程范式,允

使用C/C++调用libcurl调试消息的方式

《使用C/C++调用libcurl调试消息的方式》在使用C/C++调用libcurl进行HTTP请求时,有时我们需要查看请求的/应答消息的内容(包括请求头和请求体)以方便调试,libcurl提供了多种... 目录1. libcurl 调试工具简介2. 输出请求消息使用 CURLOPT_VERBOSE使用 C

C++实现获取本机MAC地址与IP地址

《C++实现获取本机MAC地址与IP地址》这篇文章主要为大家详细介绍了C++实现获取本机MAC地址与IP地址的两种方式,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 实际工作中,项目上常常需要获取本机的IP地址和MAC地址,在此使用两种方案获取1.MFC中获取IP和MAC地址获取

C/C++通过IP获取局域网网卡MAC地址

《C/C++通过IP获取局域网网卡MAC地址》这篇文章主要为大家详细介绍了C++如何通过Win32API函数SendARP从IP地址获取局域网内网卡的MAC地址,感兴趣的小伙伴可以跟随小编一起学习一下... C/C++通过IP获取局域网网卡MAC地址通过win32 SendARP获取MAC地址代码#i

解读静态资源访问static-locations和static-path-pattern

《解读静态资源访问static-locations和static-path-pattern》本文主要介绍了SpringBoot中静态资源的配置和访问方式,包括静态资源的默认前缀、默认地址、目录结构、访... 目录静态资源访问static-locations和static-path-pattern静态资源配置

JSON字符串转成java的Map对象详细步骤

《JSON字符串转成java的Map对象详细步骤》:本文主要介绍如何将JSON字符串转换为Java对象的步骤,包括定义Element类、使用Jackson库解析JSON和添加依赖,文中通过代码介绍... 目录步骤 1: 定义 Element 类步骤 2: 使用 Jackson 库解析 jsON步骤 3: 添