C++学习进阶篇之类大小计算和this指针

 更新时间:2023年04月27日 09:05:01   作者:青山与你  
this是C++中的一个关键字,也是一个const指针,它指向当前对象,通过它可以访问当前对象的所有成员,下面这篇文章主要给大家介绍了关于C++学习进阶篇之类大小计算和this指针的相关资料,需要的朋友可以参考下

一、类大小计算

类的大小是只计算它的成员变量或者自定义成员,不会计算它的成员函数大小。

#include<iostream>
using namespace std;

class A1
{
public:
	void fun()
	{
		_a = 1;
		_b = 2;
	}

	int _a;
	int _b;
};

int main()
{
	cout << sizeof(A1) << endl;//8字节,为何是8字节
	
	return 0;
}

光是类中成员变量_a,_b所占的字节大小就为8了,但是类中还要成员函数,为何还是8字节 难道成员函数没有在类之中,对的
类成员函数在公共代码区的,不同对象的成员数据是私有的每个人有自己的成员数据
但是成员函数是共有的,此时别人也可以用,所有它们并不是都要有成员函数成员函数反而占用更多的空间
类定义对象之后,相当于小区房子,每个房子都有厨房,卧室(相当于数据)
但是这些房子不一定要有篮球场(相当于成员函数),如果每一个房子都有篮球场可想而知这个房子占地会有多大
其次篮球场得很多人来玩篮球涩,不可能每次都是你一个人玩,无不无聊,得喊朋友邻居一起,这些朋友邻居也能玩
还不如将其修在小区中心,这样大大节省了空间 而类它的成员函数就是放在公共代码区的,所以不占用类空间大小

空类或者只有成员函数时,类大小?

1字节大小,它占个位,虽然一个房子它可能什么都么有但是它难道就不是房子了吗,起码那块地还是一个房子,理应有一个位置

#include<iostream>
using namespace std;

class A1//类中只有成员函数
{
	void fun()
	{}
};

class A2 {};//空类

int main()
{
	cout << sizeof(A1) << endl;
	cout << sizeof(A2) << endl;
	
	A1 a;
	A2 b;
	//虽然里面什么都没有但是这个位置是要在的
	cout << &a << endl;
	cout << &b << endl;
	return 0;
}

没有成员变量的类对象,需要1byte是为了占位,表示对象存在

二、this指针

作为成员函数的隐形指针,接收对象地址

#include<iostream>
using namespace std;

class day
{
public:
	void init(int year, int month, int day)
	{
		_year = year;
		_month = month;
		_day = day;
	}

	void Print()
	{
		//cout << this->_year  << '-' << this->_month << '-' << this->_day << endl;
		cout << _year << '-' <<_month << '-' << _day << endl;
	}
	int _year;
	int _month;
	int _day;
};

int main()
{
	day d1;
	day d2;
	d1.init(2023, 4, 24);
	d2.init(2022, 4, 24);

	d1.Print();
	d2.Print();

	return 0;
}

虽然两个对象初始化不同,但是我在调用打印函数时,实参可是什么都没传,都是call同一个地址,为何结果却不同?
既然类中的成员函数是放在公共代码区的,不在对象总,这个函数的地址是固定的,
那么为何两个对象调用它(函数)时所得结果会不一样(都是call同一个地址)

是因为对象在调用时编译器隐含的将对象的地址作为实参给传了过去,而函数形参隐含用this指针接收 在调用时,默认将对象地址当作实参传过去

调试程序,然后右击鼠标,点击转到反汇编,查看汇编代码

这些汇编代码是在将实参压栈

可以发现原本调用成员函数Init时只是显式传了三个实参,但是在底层汇编指令却压了四个参数,那是因为将对象的地址作为实参隐含压栈

在压栈时,会将对象的地址作为实参压栈传给它所调用的成员函数

在vs下面,对this指针传递进行了优化,将对象的地址放在寄存器(rcx)中,寄存器(rcx)存this指针的值

Print这里有隐含对象形参指针的,但是不能将其显式的写出来
不可修改this指针指向的对象,因为this是被const修饰(相当于day*const this)它的指向对象不可被修改
但是它指向对象的内容可以被修改
也可以通过this指针访问对象成员变量
虽然成员函数形参this指针不能显式写出
但是在成员函数内要访问对象中成员变量时却可以显示写出

可不可以这样访问类的成员变量?

不可这样直接访问,就算是类所以成员都是对外开放的,但是类中的成员变量只是声明了,并没有将其实例化,相当于房子,只拿着一张图纸,你就想在现实世界找到对应的房间,抱歉不可能

那么可不可以这样直接调用类的成员函数?

这种直接在类中调用它的成员函数,其实是不行的,因为这都没有定义出对象,编译器在调用Print函数时是会隐晦的将对象地址当作实参传过去,这里连对象都没有如何调用?

那么有人又会说,那我传空地址可否?在调用时,对象地址是隐晦的压栈,并不能显式的写出来,也是不行的

#include<iostream>
using namespace std;


class day
{
public:
	void init(int year, int month, int day)
	{
		_year = year;
		_month = month;
		_day = day;
	}

	void Print()
	{
		cout << this->_year  << '-' << this->_month << '-' << this->_day << endl;
	}
	int _year;
	int _month;
	int _day;
};

int main()
{
	day* p = nullptr;
	p->init(2023, 4, 24);
	p->Print();
}

p->initp调用函数init会对这个函数解引用吗?不会,因为这个函数的地址不在这个对象中,它在公共代码区 p会作为实参传给this指针
虽然定义类对象是为空的指针,但是调用类成员初始化函数,并没有报错,是因为没有访问对象中的成员变量

这样会发生错误,因为访问对象中的成员,但是this指针的值为空,对空指针解引用,这样极其危险的操作

总结

到此这篇关于C++学习进阶篇之类大小计算和this指针的文章就介绍到这了,更多相关C++类大小计算和this指针内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • 详解C++的反调试技术与绕过手法

    详解C++的反调试技术与绕过手法

    反调试技术,恶意代码会用它识别自身是否被调试,或者让调试器失效,给反病毒工程师们制造麻烦,拉长提取特征码的时间线,本章将具体总结常见的反调试基础的实现原理以及如何过掉这些反调试手段,从而让我们能够继续分析恶意代码
    2021-06-06
  • 详解C++中多态的底层原理

    详解C++中多态的底层原理

    要了解C++多态的底层原理需要我们对C指针有着深入的了解,这个在打印虚表的时候就可以见功底,所以快来跟随小编一起学习一下吧
    2022-04-04
  • C语言模拟实现库函数详解

    C语言模拟实现库函数详解

    C语言库函数是把自定义函数放到库里,是别人把一些常用到的函数编完放到一个文件里,供程序员使用,下面让我们一起来详细了解它
    2022-07-07
  • 一起来学习C语言的程序环境与预处理

    一起来学习C语言的程序环境与预处理

    这篇文章主要为大家详细介绍了C语言程序环境与预处理,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下,希望能够给你带来帮助
    2022-03-03
  • C++实现LeetCode(8.字符串转为整数)

    C++实现LeetCode(8.字符串转为整数)

    这篇文章主要介绍了C++实现LeetCode(8.字符串转为整数),本篇文章通过简要的案例,讲解了该项技术的了解与使用,以下就是详细内容,需要的朋友可以参考下
    2021-07-07
  • C字符串与C++中string的区别详解

    C字符串与C++中string的区别详解

    以下是对C字符串与C++中string的区别进行了详细的分析介绍,需要的朋友可以过来参考下
    2013-09-09
  • 基于C语言实现静态通讯录的示例代码

    基于C语言实现静态通讯录的示例代码

    这篇文章主要为大家详细介绍了如何利用C语言实现一个简单的静态通讯录,文中的示例代码讲解详细,对我们学习C语言有一定帮助,需要的可以参考一下
    2022-07-07
  • 使用C++实现工资管理中的随机教师信息生成功能

    使用C++实现工资管理中的随机教师信息生成功能

    这篇文章主要介绍了使用C++实现工资管理中的随机教师信息生成功能,想要做一个教师工资管理系统,就必须得准备好数据,但是这些数据如果用手一行一行地敲,那么工作量是非常大的,因此,我就产生了用C语言实现直接生成大量的教师基本信息的想法,需要的朋友可以参考下
    2023-05-05
  • matlab模拟退火算法单约束车间流水线调度解决实现及示例

    matlab模拟退火算法单约束车间流水线调度解决实现及示例

    这篇文章主要为大家介绍了matlab模拟退火算法求解单约束车间流水线调度的实现及示例,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步
    2022-02-02
  • linux环境下C++实现俄罗斯方块

    linux环境下C++实现俄罗斯方块

    这篇文章主要为大家详细介绍了linux环境下C++实现俄罗斯方块,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2018-06-06

最新评论