C++类对象模型、类对象的存储方式、this指针、this指针的引出、this指针的特性、C语言和C++实现Stack的对比等的介绍。

文章目录

  • 前言
  • 一、C++类对象模型
    • 1. 类对象的存储方式
    • 2. 结构体内存对齐规则
  • 二、this指针
    • 1. this指针的引出
    • 2. this指针的特性
    • 3. C语言和C++实现Stack的对比
  • 总结


前言

C++类对象模型、类对象的存储方式、this指针、this指针的引出、this指针的特性、C语言和C++实现Stack的对比等的介绍。


一、C++类对象模型

1. 类对象的存储方式

只保存成员变量,成员函数存放在公共的代码段

#include <iostream>
using namespace std;

// 类中包含成员变量和成员函数
class A1
{
public:
	void f1() {};
private:
	char _str;
	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;
}

在这里插入图片描述

结论:一个类的大小,实际就是该类中”成员变量”之和,当然要注意内存对齐
注意空类的大小,空类比较特殊,编译器给了空类一个字节来唯一标识这个类的对象。

2. 结构体内存对齐规则

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

二、this指针

1. this指针的引出

#include <iostream>
using namespace std;

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(2024, 6, 17);
	d2.Init(2022, 11, 11);
	d1.Print(); // 2024 6 17
	d2.Print(); // 2022 11 11
		

	return 0;
}
  • 有上述类对象的存储方式可知,成员函数是在公共区域中的,所以d1 和 d2 调用的是同一个函数。
  • 但是打印出的结果是不同的,编译器是如何区分的呢?

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

  • 简单来说就是,每一个成员函数在调用时,编译器自动传入一个隐藏的this指针,该指针指向调用该函数的对象。函数中对成员变量的操作都是通过该this指针访问的。

在这里插入图片描述

  • 调用函数的大致过程如上:
  • 但是this是一个关键字,是编译器自动完成传参的,不能在形参和实参中显示传递。但是在函数内部可以直接使用。如下:
    在这里插入图片描述

2. this指针的特性

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

this 指针存放在哪里?

  • this指针本质上是函数的形参,所以this指针存放在栈区中。

this 指针可以为空吗?

  • this 指针可以为空。如下:
#include <iostream>
using namespace std;

class A
{
public:
	void Print()
	{
		cout << "Print()" << endl;
	}
private:
	int _a;
};


int main()
{
	
	A* p = nullptr;
	p->Print();

	return 0;
}
  • 如上述代码,p为空指针,在调用类A的成员函数时,传入了p,即此时隐藏的this为空指针。
  • 上述代码能成功运行并打印的原因:
    类对象的成员函数是存放在公共区域中的,不存在类内部。并且,成员函数内部并没有访问成员变量,因此没有对this解引用。所以,程序可以成功运行。

#include <iostream>
using namespace std;

class A
{
public:
	void Print()
	{
		cout << _a << endl;
	}
private:
	int _a;
};


int main()
{

	A* p = nullptr;
	p->Print();

	return 0;
}

在这里插入图片描述

  • 上述代码在调用函数时传入空指针,但成员函数存放在公共区域中,this空指针无影响。
  • 但是 函数内部访问了成员变量,即对this指针进行解引用,所以会报错(空指针解引用,运行时错误)

3. C语言和C++实现Stack的对比

C语言实现Stack

#include <stdio.h>
#include <assert.h>
#include <stdlib.h>
#include <stdbool.h>

#define DEFAULT_CAPACITY 4
typedef int STDataType;
typedef struct Stack
{
	STDataType* a;
	int size;
	int capacity;
}Stack;


void StackInit(Stack* ps)
{
	assert(ps);

	ps->a = (STDataType*)malloc(sizeof(STDataType) * DEFAULT_CAPACITY);
	if (ps->a == NULL)
	{
		perror("StackInit malloc");
		return;
	}

	ps->size = 0;
	ps->capacity = DEFAULT_CAPACITY;
}


void StackDestroy(Stack* ps)
{
	assert(ps);
	free(ps->a);
	ps->a = NULL;

	ps->size = 0;
	ps->capacity = 0;
}

void StackCheckCapacity(Stack* ps)
{
	if (ps->size == ps->capacity)
	{
		STDataType* tmp = (STDataType*)realloc(ps->a, sizeof(STDataType) * (ps->capacity) * 2);
		if (tmp == NULL)
		{
			perror("StackCheckCapacity realloc");
			return;
		}

		ps->a = tmp;
		tmp = NULL;
		ps->capacity *= 2;
	}
}


void StackPush(Stack* ps, STDataType x)
{
	assert(ps);

	StackCheckCapacity(ps);

	ps->a[ps->size] = x;
	ps->size++;
}

bool StackEmpty(Stack* ps)
{
	assert(ps);

	return (ps->size == 0);

}

void StackPop(Stack* ps)
{
	assert(ps && ps->size);

	ps->size--;
}

STDataType StackTop(Stack* ps)
{
	assert(ps && ps->size);

	return (ps->a[ps->size - 1]);
}

int StackSize(Stack* ps)
{
	assert(ps);

	return ps->size;
}




int main()
{
	Stack st;
	StackInit(&st);

	StackPush(&st, 1);
	StackPush(&st, 2);
	StackPush(&st, 3);
	StackPush(&st, 4);
	StackPush(&st, 5);

	printf("%d\n", StackSize(&st));

	while (!StackEmpty(&st))
	{
		printf("%d ", StackTop(&st));
		StackPop(&st);
	}
	printf("\n");

	printf("%d\n", StackSize(&st));


	StackDestroy(&st);


	return 0;
}

在这里插入图片描述

可以看到,在用C语言实现时,Stack相关操作函数有以下共性:

  • 每个函数的第一个参数都是Stack*
  • 函数中必须要对第一个参数检测,因为该参数可能会为NULL
  • 函数中都是通过Stack*参数操作栈的
  • 调用时必须传递Stack结构体变量的地址

结构体中只能定义存放数据的结构,操作数据的方法不能放在结构体中,即数据和操作数据
的方式是分离开的
,而且实现上相当复杂一点,涉及到大量指针操作,稍不注意可能就会出
错。


C++实现Stack

#include <iostream>
#include <assert.h>
using namespace std;

typedef int STDataType;
typedef struct Stack
{
public:
	// 初始化栈
	void Init(int capacity)
	{
		_a = (STDataType*)malloc(sizeof(STDataType) * capacity);
		if (_a == nullptr)
		{
			perror("Init malloc");
			return;
		}

		_capacity = capacity;
		_size = 0;
	}

	// 销毁栈
	void Destroy()
	{
		free(_a);
		_a = nullptr;
		_size = 0;
		_capacity = 0;
	}

	// 插入数据
	void Push(STDataType x)
	{
		if (_size == _capacity)
		{
			STDataType* tmp = (STDataType*)realloc(_a, sizeof(STDataType) * 2 * _capacity);
			if (tmp == nullptr)
			{
				perror("Push realloc");
				return;
			}

			_a = tmp;
			_capacity *= 2;
		}

		_a[_size] = x;
		_size++;
	}

	// 判断是否为空
	bool Empty()
	{
		return (_size == 0);
	}

	// 出栈顶元素
	void Pop()
	{
		assert(!Empty());
		_size--;
	}

	// 获得栈顶元素
	STDataType Top()
	{
		return _a[_size - 1];
	}

	// 获得栈的大小
	int Size()
	{
		return _size;
	}

private:
	STDataType* _a;
	int _size;
	int _capacity;
}Stack;

int main()
{
	Stack st;
	st.Init(4);

	st.Push(1);
	st.Push(2);
	st.Push(3);
	st.Push(4);
	st.Push(5);


	cout << st.Top() << endl;
	cout << st.Size() << endl;

	while (!st.Empty())
	{
		cout << st.Top() << " ";
		st.Pop();
	}
	cout << endl;

	cout << st.Size() << endl;

	st.Pop();

	st.Destroy();
	

	return 0;
}

在这里插入图片描述

C++中通过类可以将数据 以及 操作数据的方法进行完美结合,通过访问权限可以控制那些方法在
类外可以被调用,即封装
,在使用时就像使用自己的成员一样,更符合人类对一件事物的认知。
而且每个方法不需要传递Stack*的参数了,编译器编译之后该参数会自动还原,即C++中 Stack *
参数是编译器维护的,C语言中需用用户自己维护


总结

C++类对象模型、类对象的存储方式、this指针、this指针的引出、this指针的特性、C语言和C++实现Stack的对比等的介绍。

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

【论文阅读笔记】PA-SAM: Prompt Adapter SAM for High-Quality Image Segmentation

1.论文介绍 PA-SAM: Prompt Adapter SAM for High-Quality Image Segmentation PA-SAM&#xff1a;用于高质量图像分割的提示适配器SAM 2024年 ICME Paper Code 2.摘要 Segment Anything Model&#xff0c;SAM在各种图像分割任务中表现出了优异的性能。尽管SAM接受了超过10亿…

揭秘“湖仓一体”——Flink+Paimon+StarRocks,打造实时分析新纪元

1.湖仓一体 数据湖仓是 Flink 流批一体发挥重要作用的场景,使用 Flink Paimon starRocks 来构建湖仓一体数据分析. Apache Paimon 是一个专为实时数据处理而设计的湖表格式&#xff0c;它最大的亮点是使用了 LSM Tree 技术。与 Hudi 相比&#xff0c;Paimon 在更新插入&…

中国星网时空信息集团春招Offer面经

本文介绍2024届春招中&#xff0c;中国卫星网络集团有限公司下属中国时空信息集团有限公司中&#xff0c;业务助理岗位1场面试的基本情况、提问问题等。 2024年04月投递了中国卫星网络集团有限公司下属中国时空信息集团有限公司中的业务助理岗位&#xff0c;所属部门为运营中心…

第9章 类

第9章 类 9.1 创建和使用类9.1.1 创建 Dog 类9.1.2 根据类创建实例 9.2 使用类和实例9.2.1 Car 类9.2.2 给属性指定默认值9.2.3 修改属性的值 9.3 继承9.3.1 子类的方法__init__()9.3.2 给子类定义属性和方法9.3.3 重写父类的方法9.3.4 将实例用作属性9.3.5 模拟实物 9.4 导入类…

短URL服务设计

引言 在营销系统里&#xff0c;为了增加系统的活跃用户数&#xff0c;经常会有各种各样的营销活动。这类活动几乎都是为了充分利用存量用户的价值&#xff0c;促使他们分享产品或App以达到触达到更多用户的目的。又或者是出于营销目的&#xff0c;群发优惠券触达短信这种场景。…

充电学习—3、Uevent机制和其在android层的实现

sysfs 是 Linux userspace 和 kernel 进行交互的一个媒介。通过 sysfs&#xff0c;userspace 可以主动去读写 kernel 的一些数据&#xff0c;同样的&#xff0c; kernel 也可以主动将一些“变化”告知给 userspace。也就是说&#xff0c;通过sysfs&#xff0c;userspace 和 ker…

欣九康诊疗系统助力诊所向数字化转型

数字化已经成为各行各业转型的重点方向&#xff0c;而为了不被时代所淘汰&#xff0c;医疗机构也势必要紧跟潮流&#xff0c;本人作为门诊部的负责人深知医疗机构要想实现数字化转型那么拥有一款便捷实用的医疗平台是必不可少的&#xff0c;近几年&#xff0c;随着国家大力支持…

Ubuntu 在线或离线安装docker

查看自己的ubuntu版本 在终端中执行以下命令&#xff1a; lsb_release -a 终端中的复制粘贴&#xff1a; ctrl shift c ctrl shifr v 在线安装docker&#xff08;不需要外网&#xff09;: 命令行安装&#xff1a;Ubuntu Docker -- 从入门到实践 看完…

Ollama:本地部署大模型 + LobeChat:聊天界面 = 自己的ChatGPT

本地部署大模型 在本地部署大模型有多种方式&#xff0c;其中Ollama方式是最简单的&#xff0c;但是其也有一定的局限性&#xff0c;比如大模型没有其支持的GGUF二进制格式&#xff0c;就无法使用Ollama方式部署。 GGUF旨在实现快速加载和保存大语言模型&#xff0c;并易于阅读…

香港Web3时代:比特币可以成为「收益性资产」吗?

原文标题&#xff1a;《CAN BITCOIN BE A PRODUCTIVE ASSET?》撰文&#xff1a;Pascal Hgli编译&#xff1a;Chris&#xff0c;Techub News本文来源香港Web3媒体 Techub News 比特币正在经历一场大的变化&#xff0c;人们对其性质有不同的看法。有些人将其视为日常交易的货币…

ANSYS EMC解决方案与经典案例

EMC问题非常复杂&#xff0c;各行各业都会涉及&#xff0c;例如航空、航天、船舶、汽车、火车、高科技、物联网、消费电子。要考虑EMC的对象很多&#xff0c;包含整个系统、设备、PCB、线缆、电源、芯片封装。而且技术领域覆盖广&#xff0c;涉及高频问题、低频问题&#xff1b…

AI大模型系统从入门到精通,看这一篇就够了

前言 2023 年&#xff0c;人工智能发展达到新的里程碑。自 GPT 系列和 LLaMA 系列等大规模语言模型及应用问世以来&#xff0c;AI 内部技术突飞猛进&#xff0c;能力迅速超越以往。这些“超级 AI 助手”看似便捷强大&#xff0c;但其背后复杂原理及潜在影响值得深入思考。 这些…

充电学习—5、healthed 电池服务

1、healthed服务监听接收内核kernel的电池事件&#xff0c;然后上传数据给framware层的batterysevice&#xff0c;BatteryService计算电池的电量&#xff0c;显示&#xff0c;绘制动画等 android电池系统框架&#xff1a; 2、healthd服务入口&#xff1a;android/system/cor…

本地安装nightingale监控分析服务并发布公网详细流程

文章目录 前言1. Linux 部署Nightingale2. 本地访问测试3. Linux 安装cpolar4. 配置Nightingale公网访问地址5. 公网远程访问Nightingale管理界面6. 固定Nightingale公网地址 前言 本文主要介绍如何在本地Linux系统部署 Nightingale 夜莺监控并结合cpolar内网穿透工具实现远程…

怎么把两个音频合成一个?将两个音频合成一个的四种方法

怎么把两个音频合成一个&#xff1f;在当今数字化的时代&#xff0c;音频处理已经成为我们生活中不可或缺的一部分。有时候&#xff0c;我们会希望将两段音频合成为一个&#xff0c;无论是为了制作音乐混音、创作声音效果&#xff0c;还是为了编辑播客节目或视频配音。合成音频…

Qt第三方库QHotKey设置小键盘数字快捷键

一、看了一圈没有找到可以设置小键盘的情况。 这两天在研究快捷键的使用。发现qt的里的快捷键不是全局的。找了两个第三方快捷键QHotKey&#xff0c;还有一个QxtGlobalShortcut。但是这两个都不能设置小键盘的数字。 比如QKeySequenceEdit &#xff08;Ctrl1&#xff09; 这个…

springboot小型超市商品展销系统-计算机毕业设计源码01635

摘 要 科技进步的飞速发展引起人们日常生活的巨大变化&#xff0c;电子信息技术的飞速发展使得电子信息技术的各个领域的应用水平得到普及和应用。信息时代的到来已成为不可阻挡的时尚潮流&#xff0c;人类发展的历史正进入一个新时代。在现实运用中&#xff0c;应用软件的工作…

onnx基本概念

onnx基本概念 参考 文章目录 onnx基本概念Input, Output, Node, Initializer, AttributesSerialization with protobuf元数据List of available operators and domains支持的类型Opset版本Subgraphs, tests and loopsExtensibilityFunctionsShape (and Type) Inferencetools O…

Fiddler抓包工具介绍

下载 下载:Web Debugging Proxy and Troubleshooting Tools|Fiddler 进去要填一个表 汉化版 百度网盘 请输入提取码 提取码&#xff1a;xq9t 下载过附件之后分别把两个文件 点开fiddler就ok了 配置https fiddler要想抓到https包(解密的),点击tools->options勾选三个对…

数据结构之“双向链表”

前言 前面我们介绍了单向链表&#xff0c;我们这里的双向链表是为了弥补单向链表只能从头节点开始单向遍历&#xff0c;插入和删除节点时需要更多的操作&#xff0c;因为无法直接访问前一个节点。 目录 前言 一、双向链表的结构 二、实现双向链表 2.1符号定义 2.2节点创…