[C++] 类与对象(上)

目录

1、前言

2、类的引入

3、类的定义

3.1 类的两种定义方式

4、类的访问限定符

5、类的作用域

6、类的实例化

7、类对象模型

7.1 内存对齐规则

7.1 类对象的存储方式

8、this指针

8.1 this指针的特性

8.2 this指针是否可以为空


1、前言

C语言是面向过程的,关注的是过程,分析出求解问题的步骤,通过函数调用逐步解决问题。
C++是基于面向对象的,关注的是对象,将一件事情拆分成不同的对象,靠对象之间的交互完成。

我们来举列子说明一下:

比如蒸米饭这件事,C语言关注的就是,淘米 -> 加水 -> 加米 -> 起锅 -> 烧锅 -> 水沸腾 -> 放入盛有生米的盆 -> 烧锅 -> 拿出;C++在这里关注的是对象:人、米、水、电饭煲,人不需要知道电饭煲如果工作的,只需要四个对象之间交互完成任务就可以。

2、类的引入

C语言结构体(struct)中只能定义变量。

而在C++中,新增加了玩法,结构体内不仅可以定义变量,也可以定义函数

举例:

struct Person
{void personInit(){cout << "void personInit()" << endl;}int _age;char _name[20];
};
int main()
{Person p;p.personInit();return 0;
}

上面结构体的定义,在C++中更喜欢用class来代替。

3、类的定义

class className
{
// 类体:由成员函数和成员变量组成
}; // 一定要注意后面的分号

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

3.1 类的两种定义方式

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

需注意:成员函数如果在类中定义,编译器可能会将其当成内联函数处理。

class Person
{
public:void PersonInit(){cout << "void PersonInit()" << endl;}private:int _age;char _name[20];
};

3.1.2 类声明放在.h文件中,成员函数定义放在.cpp文件中
注意:成员函数名前需要加类名::

//Person.h
class Person
{
public:void PersonInit();private:int _age;char _name[20];
};//Person.c
#include "Person.h"
void Person::PersonInit()
{cout << "void Person::PersonInit()" << endl;
}

一般情况下,更期望采用第二种方式。

注意:一般在声明成员变量的时候成员变量前加_(下划线),为区分成员函数的形参(只要能区分就可以,前家下划线是我的一种方式,谷歌C++规范一般喜欢后加_)。

如果成员变量没有特殊标记时,当在成员函数内用到成员变量,并为其赋值的时候,函数会采用局部优先的原理,将自己赋给自己,这样就达不到预期的效果。

4、类的访问限定符

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

【访问限定符说明】

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

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

5、类的作用域

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

class Person
{
public:void PersonInit();//声明private:int _age;char _name[20];
};//初始化定义的时候函数名前加 Person::,表明PersonInit是Person类域的
void Person::PersonInit()//定义
{cout << "void Person::PersonInit()" << endl;
}

6、类的实例化

用类类型创建对象的过程,称为类的实例化。

在实例化之前,定义出来的类是不占空间的。
例如:

class Person
{
public:void PersonInit(){cout << "void PersonInit()" << endl;}private:int _age;		//声明char _name[20];
};int main()
{Person::_age = 20;//定义开空间Person p;return 0;
}

在没有实例化p的时候,定义的 class Person 类不会开辟空间。

实例化出了 p 对象,这时 p 是占用实际的空间的,存储了成员变量。(这里实例化后,p也不能直接用_age,因为是私有的,后面会讲如何赋值,打印)。

类就像是设计图。

类实例化出对象就像现实中使用设计图盖房子。

在没有盖房子之前,这块区域没有占空间。

实例化就是按图盖房子,这时就占用了空间。

我们可以使用设计图盖多个房子,这些房子都占空间

7、类对象模型

如何计算类对象的大小呢?

例如:

class A
{
public:void AInit(int a, int b){cout << "void AInit(int a, int b)" << endl;}private:int _a;int _b;
};
class B
{private:int _a;int _b;
};
class C
{};
int main()
{cout << "类A的大小:" << sizeof(A) << endl;cout << "类B的大小:" << sizeof(B) << endl;cout << "类C的大小:" << sizeof(C) << endl;return 0;
}

运行结果:

总结:

1、成员函数不算在类的大小中;

2、类的大小只与成员变量有关,并遵循结构体对齐规则;

3、空类的大小为1字节(不存储数据,只是占位,表示对象存在过)。

7.1 内存对齐规则

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

7.1 类对象的存储方式

为什么类中的成员函数不占空间?那成员函数是存在哪里呢?

实例化后的类中只存储类成员变量

成员函数保存在公共的代码段。

我们画图来理解一下:

8、this指针

我们这里写一个日期类来看一下:

class Date
{
public:void Init(int year, int month, int day){_year = year;_month = month;_day = day;}private:int _year;int _month;int _day;
};int main()
{Date d1;d1.Init(2023, 7, 30);return 0;
}

这里看似 Init 函数只有三个形参

调用的时候传了三个参数

实际上,这里还隐含了一个 this 指针。

我们这里画图来看一下:

这里对成员变量赋值的时候,前后都会加一个 this-> 来接引用访问。

8.1 this指针的特性

1. this指针的类型:类类型* const,即成员函数中,不能给this指针赋值。
2. 只能在“成员函数”的内部使用。
3. this指针本质上是“成员函数”的形参,当对象调用成员函数时,将对象地址作为实参传递给this形参。所以对象中不存储this指针
4. this指针是“成员函数”第一个隐含的指针形参,一般情况由编译器通过ecx寄存器自动传递,不需要用户传递。

因此,我们写的时候就不可以这样写:

class Date
{
public:void Init(Date* this, int year, int month, int day){_year = year;_month = month;_day = day;}private:int _year;int _month;int _day;
};int main()
{Date d1;d1.Init(&d1, 2023, 7, 30);return 0;
}

this指针隐含着,如果我们自己加上就是错误的。

this在实参和形参的位置上不能显示写

但是在类里面可以显示的用

如下:

class Date
{
public://this在实参和形参的位置上不能显示写//但是在类里面可以显示的用void Init(int year, int month, int day){this->_year = year;this->_month = month;this->_day = day;}private:int _year;int _month;int _day;
};int main()
{Date d1;d1.Init(2023, 7, 30);return 0;
}

8.2 this指针是否可以为空

我们来看下面几段代码:

class Person
{
public:void PersonInit(){cout << "void PersonInit()" << endl;}private:int _age;		//声明char _name[20];
};int main()
{Person* p = nullptr; //初始化为空指针p->PersonInit();return 0;
}

运行结果:

这里我们就会产生疑问,p是空指针为什么可以解引用呢?还是正常运行。

这里对于函数定义在类里面且短小,编译器会当作内联函数,直接展开,并不会解引用;

而如果声明与定义分离或者编译器不将其当作内联函数,就是call Init函数(调用函数)的地址,也不是解引用。

我们继续看:

class Person
{
public:void PersonInit(){cout << "void PersonInit()" << endl;}//private:int _age;		//声明char _name[20];
};int main()
{Person* p = nullptr; //初始化为空指针p->PersonInit();p->_age = 1;return 0;
}

这就会导致运行崩溃,对空指针的内容进行解引用。

我们接着上面看:

class Person
{
public:void PersonInit(){cout << _age << endl;}//private:int _age;		//声明char _name[20];
};int main()
{Person* p = nullptr; //初始化为空指针p->PersonInit();return 0;
}

这里在调用Init函数的时候,函数里面产生了解引用,但是this是空指针,这里就会运行崩溃。

空指针不会编译错误,只会导致运行崩溃。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/16159.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

DUBBO服务多网卡,服务调用失败

如果服务器是多网卡的&#xff0c;比如安装了docker&#xff0c;有一个docker虚拟网卡&#xff0c;一个实体网卡eth0&#xff0c;当我们运行springboot应用后&#xff0c;dubbo注入到zk的地址是 docker虚拟网卡的地址172网段&#xff0c;而不是实际内网地址192网段&#xff0c;…

类的封装和包(JAVA)

封装 所有的OOP语言都会有三个特征&#xff1a; 封装&#xff1b;继承&#xff1b;多态。 本篇文章会为大家带来有关封装的知识。 在我们日常生活中可以看到电视就只有那么几个按键&#xff08;开关&#xff0c;菜单……&#xff09;和一些接口&#xff0c;而而我们通过这些东…

【计算机视觉|人脸建模】SOFA:基于风格、由单一示例的2D关键点驱动的3D面部动画

本系列博文为深度学习/计算机视觉论文笔记&#xff0c;转载请注明出处 标题&#xff1a;SOFA: Style-based One-shot 3D Facial Animation Driven by 2D landmarks 链接&#xff1a;SOFA: Style-based One-shot 3D Facial Animation Driven by 2D landmarks | Proceedings of …

jmeter压力测试指标解释

目录 RT(response time) Throughput 吞吐量 并发用户数 QPS (query per seconds) TPS (transition per seconds) PV和UV 聚合报告&#xff1a; RT(response time) 什么是RT? RT就是指系统在接收到请求和做出相应这段时间跨度 但是值得一提的是RT的值越高,并不真的就能…

什么是云原生和 CNCF?

一、CNCF简介 CNCF&#xff1a;全称Cloud Native Computing Foundation&#xff08;云原生计算基金会&#xff09;&#xff0c;成立于 2015 年 12 月 11 日&#xff0c;是一个开源软件基金会&#xff0c;它致力于云原生&#xff08;Cloud Native&#xff09;技术的普及和可持续…

Klipper seria.c 文件代码分析

一. 前言 Klipper 底层硬件的串口模块程序写的是否正确是决定下位机与上位机能否正常通信的前提&#xff0c;如果这个文件的驱动没写好&#xff0c;那上位机控制下位机就无从谈起&#xff0c;更无法通过上位机去验证下位机程序的正确性。 本篇博文将详细解析 Klipper src 文件夹…

809协议

809协议 目录概述需求&#xff1a; 设计思路实现思路分析1.809协议数据流——链路管理类 参考资料和推荐阅读 Survive by day and develop by night. talk for import biz , show your perfect code,full busy&#xff0c;skip hardness,make a better result,wait for change,…

在idea中添加try/catch的快捷键

在idea中添加try/catch的快捷键 在idea中添加try/catch的快捷键 ctrlaltt 选中想被try/catch包围的语句&#xff0c;同时按下ctrlaltt&#xff0c; 出现下图 选择try/catch即可。

Elasticsearch搜索引擎系统入门

目录 【认识Elasticsearch】 Elasticsearch主要应用场景 Elasticsearch的版本与升级 【Elastic Stack全家桶】 Logstash Kibana Beats Elasticsearch在日志场景的应用 Elasticsearch与数据库的集成 【安装Elasticsearch】 安装插件 安装Kibana 安装Logstash 【认…

C# 2的幂

231 2的幂 给你一个整数 n&#xff0c;请你判断该整数是否是 2 的幂次方。如果是&#xff0c;返回 true &#xff1b;否则&#xff0c;返回 false 。 如果存在一个整数 x 使得 n 2x &#xff0c;则认为 n 是 2 的幂次方。 示例 1&#xff1a; 输入&#xff1a;n 1 输出&a…

【图论】三种中心性 —— 特征向量、katz 和 PageRank

维基百科&#xff1a;在图论和网络分析中&#xff0c;中心性指标为图中相应网络位置的节点分配排名或数值。中心性这一概念最初起源于社交网络分析&#xff0c;因此很多衡量中心性的术语也反映了其社会学背景。 不同中心性指标对 “重要” 的衡量方式不同&#xff0c;因此适用于…

惊喜!1行Python代码,瞬间测你工作量,分享一个统计代码行数的神器

大家好&#xff0c;这里是程序员晚枫。 **你想不想知道一个项目中&#xff0c;自己写了多少行代码&#xff1f;**我用今天的工具统计了一下开源项目&#xff1a;python-office的代码行数&#xff0c;竟然有21w行&#xff01; 我们一起看一下怎么用最简单的方法&#xff0c;统…

mac下安装vue cli脚手架并搭建一个简易项目

目录 1、确定本电脑下node和npm版本是否为项目所需版本。 2、下载vue脚手架 3、创建项目 1、下载node。 如果有node&#xff0c;打开终端&#xff0c;输入node -v和npm -v , 确保node和npm的版本&#xff0c;(这里可以根据自己的需求去选择&#xff0c;如果对最新版本的内容有…

IO进程线程day3(2023.7.31)

一、Xmind整理&#xff1a; 文件描述符概念&#xff1a; 二、课上练习&#xff1a; 练习1&#xff1a;用fread和fwrite实现文件拷贝 #include <stdio.h> #include <string.h> #include <stdlib.h> #include <head.h> int main(int argc, const char…

什么叫前后端分离?为什么需要前后端问题?解决了什么问题?

单体架构出现的问题 引出&#xff1a;来看一个单体项目架构的结构 通过上述可以看到单体架构主要存在以下几点问题&#xff1a; 开发人员同时负责前端和后端代码开发&#xff0c;分工不明确开发效率低前后端代码混合在一个工程中&#xff0c;不便于管理对开发人员要求高(既会前…

千元内合板和单板吉他怎么选?SAGA萨伽SF600和VEAZEN费森CLR300怎么样?哪一款更适合初学者入门使用!【吉他评测】

对于预算不多的朋友&#xff0c;在选购前翻阅查询很多资料&#xff0c;吉他材质、桶型和尺寸等等疑问&#xff0c;不知道怎么选&#xff0c;无从下手&#xff0c;还容易遇到烧火棍&#xff0c;在这里介绍这两款VEAZEN费森CLR300&#xff08;单板&#xff09;和SAGA萨伽SF600&am…

vuejs源码阅读之代码生成器

代码生成器是模版编译的最后以后&#xff0c;它的作用是将AST转换成渲染函数中的内容&#xff0c;这个内容可以称为代码字符串。 代码字符串可以被包装在函数中执行&#xff0c;这个函数就是我们通常说的渲染函数。 渲染函数被执行之后&#xff0c;可以生成一份VNode&#xf…

分治法 Divide and Conquer

1.分治法 分治法&#xff08;Divide and Conquer&#xff09;是一种常见的算法设计思想&#xff0c;它将一个大问题分解成若干个子问题&#xff0c;递归地解决每个子问题&#xff0c;最后将子问题的解合并起来得到整个问题的解。分治法通常包含三个步骤&#xff1a; 1. Divid…

【Python系列】Python基础语法轻松入门—从变量到循环

目录 写在前面 语法介绍 变量 数据类型 整数 浮点数 字符串 列表 元组 字典 运算符 算术运算符 比较运算符 逻辑运算符 条件语句 循环语句 图书推荐 图书介绍 参与方式 中奖名单 写在前面 Python 是一种高级、解释型的编程语言&#xff0c;具有简单易学…

华为数通HCIP-IGMP(网络组管理协议)

IGMP&#xff08;网络组管理协议&#xff09; 作用&#xff1a;维护、管理最后一跳路由器以及组播接收者之间的关系&#xff1b; 应用&#xff1a;最后一跳路由器以及组播接收者之间&#xff1b; 原理&#xff1a;当组播接收者需要接收某个组别的流量时&#xff0c;会向最后…