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

news2024/11/22 14:18:15

文章目录

  • 前言
  • 一、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.coloradmin.cn/o/1831323.html

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

相关文章

如何在招聘中开始使用AI?

在人工智能时代&#xff0c;招聘团队面临着“用更少的钱做更多的事情”的压力&#xff1a;用更少的钱和更少的团队。根据一项调查&#xff0c;58%的受访者认为&#xff0c;“提高我们助教团队的效率&#xff0c;降低成本”是明年招聘职位的首要任务。在招聘中使用人工智能是提高…

Excel 如何复制单元格而不换行

1. 打开excle, sheet1右键单击>查看代码>插入>模块 输入代码 Sub CopyText() Updated by NirmalDim xAutoWrapper As ObjectSet xAutoWrapper New DataObject or GetObject("New:{1C3B4210-F441-11CE-B9EA-00AA006B1A69}")xAutoWrapper.SetText ActiveC…

东理咨询交流论坛系统

开头语&#xff1a;你好呀&#xff0c;我是计算机学长猫哥&#xff01;如果有相关需求&#xff0c;文末可以找到我的联系方式。 开发语言&#xff1a;Java 数据库&#xff1a;MySQL 技术&#xff1a;JSP技术、B/S架构 工具&#xff1a;MyEclipse 系统展示 首页 管理员功能…

如何利用TikTok矩阵源码实现自动定时发布和高效多账号管理

在如今社交媒体的盛行下&#xff0c;TikTok已成为全球范围内最受欢迎的短视频平台之一。对于那些希望提高效率的内容创作者而言&#xff0c;手动发布和管理多个TikTok账号可能会是一项繁琐且耗时的任务。幸运的是&#xff0c;通过利用TikTok矩阵源码&#xff0c;我们可以实现自…

docker hub仓库被禁用,镜像加速器站点替换

整理 站点整理之前用的daemon.json,现更改镜像加速地址替换自己的docker加速器daemon.json前面加https:// 站点整理 之前用的daemon.json,现更改镜像加速地址 vim /etc/docker/daemon.json{"registry-mirrors": ["https://4xgbe4ey.mirror.aliyuncs.com",…

利用原生HTML + CSS + JS实现歌词滚动

对于很多音乐APP&#xff0c;都有这么一个功能&#xff0c;就是根据歌曲的进度来控制对应的歌词滚动&#xff0c;如下图所示&#xff1a; 大概这样的效果&#xff0c;我此次是使用原生的HTMLCSSJS来实现的&#xff0c;以下是具体的实现过程。 1. 数据获取与处理 对于数据来源&…

揭秘低代码平台:解锁表尾统计方案

前言 在现代Web应用中&#xff0c;数据表格是常见的界面元素之一&#xff0c;用于展示和管理大量的数据。而vxe-table作为Vue.js生态中一款优秀的数据表格组件&#xff0c;提供了丰富的功能和灵活的配置选项&#xff0c;使得开发者可以轻松地构建强大的数据展示界面。 然而&…

普林斯顿大学教授终于把算法整理成图解笔记

普林斯顿大学教授终于把算法整理成图解笔记了&#xff01;&#xff01;&#xff01; 这些年虽然学到的编程知识越来越多&#xff0c;但是我对算法却始终没搞明白&#xff0c;直到偶然间看到这份笔记&#xff0c;我才认识到这些概念是多么简单。 对于很多刚入门的小伙伴来说&am…

充电学习—1、psy框架梳理

一、linux充电驱动代码框架&#xff1a; APP 层 该部分属于电量上报的最后的环节。其主要工作是&#xff1a;监听系统广播并对 UI 作出相应更新&#xff0c;包括电池电量百分比&#xff0c;充电状态&#xff0c;低电提醒&#xff0c;led 指示灯&#xff0c;异常提醒等FrameWork…

字符串专题详解

目录 字符串hash进阶 KMP算法 next数组 KMP算法 KMP算法优化 字符串hash进阶 字符串hash是指将一个字符串S映射为一个整数&#xff0c;使得该整数可以尽可能唯一地代表字符串S。那么在一定程度上&#xff0c;如果两个字符串转换成的整数相等&#xff0c;就可以认为这两个…

Typora—适用于 Mac 和 Win 系统的优秀 Markdown 文本编辑器

Typora 是一款适用于 Mac 和 Win 系统的优秀 Markdown 文本编辑器&#xff0c;它以其简洁易用的界面和强大的功能受到了众多用户的喜爱。 首先&#xff0c;Typora 的界面设计非常简洁直观&#xff0c;没有过多繁杂的菜单和按钮&#xff0c;让用户能够专注于写作本身。它采用实时…

【计算机毕业设计】235基于微信小程序点餐系统

&#x1f64a;作者简介&#xff1a;拥有多年开发工作经验&#xff0c;分享技术代码帮助学生学习&#xff0c;独立完成自己的项目或者毕业设计。 代码可以私聊博主获取。&#x1f339;赠送计算机毕业设计600个选题excel文件&#xff0c;帮助大学选题。赠送开题报告模板&#xff…

抖音用户新作品监控助手,第一时间获取博主作品信息。

声明&#xff1a; 本文以教学为基准、本文提供的可操作性不得用于任何商业用途和违法违规场景。本人对任何原因在使用本人中提供的代码和策略时可能对用户自己或他人造成的任何形式的损失和伤害不承担责任。包含关注&#xff0c;点赞等 抖音新作品监控助手系统是一个功能强大的…

华火新能源集成灶评测:创新与品质的融合

在厨房电器的不断推陈出新中&#xff0c;华火新能源集成灶以其独特的魅力进入了人们的视野。今天&#xff0c;我们就来深入评测这款备受关注的产品——华火新能源集成灶 一、华火新能源集成灶的创新与环保 首先&#xff0c;我们先来探讨新能源集成灶的整体表现。华火新能源集成…

Java健身私教服务师傅小程序APP源码(APP+小程序+公众号+H5)

强身健体&#xff0c;私人定制的健身之旅 &#x1f3cb;️ 引言&#xff1a;探索私人健身新纪元 在现代都市的快节奏生活中&#xff0c;越来越多的人开始注重身体健康和健身塑形。然而&#xff0c;传统的健身房模式可能无法满足每个人的个性化需求。这时&#xff0c;一款名为…

如何利用V-Ray优化渲染效果——渲染100邀请码【7788】

3dmax效果图云渲染--渲染100 以3ds Max2025、VR6.2、CR11.2等最新版本为基础&#xff0c;兼容fp、acescg等常用插件&#xff0c;同时UT滤镜等参数也得到了同步支持。 注册填邀清码[7788]领30元礼包和免费渲染券哦~ 第一步&#xff1a;初次渲染 打开场景并将V-Ray设置为当前渲染…

基于复旦微FM33FG065A+OSRAM RGBi车载氛围灯方案

应用场景&#xff1a; 汽车内饰照明&#xff1a;这种方案可以用于改善汽车内部的氛围&#xff0c;提升驾驶舒适度和乘客体验。可以通过调节灯光颜色和亮度来营造不同的氛围&#xff0c;比如温馨的暖色调或者清爽的冷色调。 个性化定制&#xff1a;汽车制造商或者车主可以根据个…

kubesphere踩过的坑,持续更新....

踩过的坑 The connection to the server lb.kubesphere.local:6443 was refused - did you specify the right host… 另一篇文档中 dashboard 安装 需要在浏览器中输入thisisunsafe,即可进入登录页面 ingress 安装的问题 问题描述&#xff1a; 安装后通过命令 kubectl g…

Python实现连连看11

(2)isOneCornerLink()函数 isOneCornerLink()函数判断图片是否是单拐点连通。判断是否是单拐点连通有两种方式,如图14所示。 图14 判断单拐点连通的两种方式 从图14中可以看出,判断两张图片是否是单拐点连通,实际上就是判断图中红点与绿点所在图片是否分别与这两张图是直…

电商平台数据的认知与深度理解

随着信息技术的迅猛发展&#xff0c;电商平台已成为现代社会商业活动的重要舞台。在这个舞台上&#xff0c;数据不仅是交易的记录&#xff0c;更是企业决策的依据、用户行为的镜子和市场变化的晴雨表。本文将从多个维度对电商平台数据进行全面且深入的认知和理解。 一、数据的…