C++ 类和对象(A)

news/2024/8/22 18:25:20 标签: c++, 开发语言, 笔记

一、类与对象的初步认识

1.类是对象的抽象,而对象是类的具体实例。

类是抽象的,不占用内存;而对象是具体的,占用存储空间。

2.面向过程与面向对象

C语言是面向过程的,关注的是过程中的数据与方法。
C++是面向对象的,关注的是对象’的属性与功能。

二、类的定义

1、类定义格式

1.class为定义类的关键字,Text为类的名字,{ }中为类的主体,注意类定义结束时后面分号不能省略。类体中内容称为类的成员:类中的变量称为类的属性或成员变量; 类中的函数称为类的方法或者成员函数。
2.为了区分成员变量,一般习惯上成员变量会加一个特殊标识,如成员变量前面或者后面加_ 或者 m开头,注意C++中这个并不是强制的,只是一些惯例。
3.C++中struct也可以定义类,C++兼容C中struct的用法,同时struct升级成了类,明显的变化是struct中可以定义函数,一般情况下我们还是推荐用class定义类。
4.定义在类里面的成员函数默认为inline。

#include<iostream>
using namespace std;

class Text
{
public:
	void Print()
	{
		n = 1;
		cout << n << endl;
	}
	void Init(int year, int month, int day)
	{
	_year = year;
	_month = month;
	_day = day;
	}
private:
	int n;
	// 为了区分成员变量,⼀般习惯上成员变量
	// 会加⼀个特殊标识,如_ 或者 m开头
	int _year;  // year_ m_year
	int _month;
	int _day;
};
int main()
{
	Text d;
	d.Init(2024, 3, 31);
	d.Print();
	return 0;
}
#include<iostream>
using namespace std;
// C++升级struct升级成了类
// 1、类⾥⾯可以定义函数
// 2、struct名称就可以代表类型
// C++兼容C中struct的⽤法
typedef struct ListNodeC
{
	struct ListNodeC* next;
	int val;
}LTNode;
// 不再需要typedef,ListNodeCPP就可以代表类型
struct ListNodeCPP
{
	void Init(int x)
	{
		next = nullptr;
		val = x;
	} 
	ListNodeCPP* next;
	int val;
};
int main()
{
	return 0;
}

2、访问限定符

1.C++⼀种实现封装的方式,用类将对象的属性与方法结合在⼀块,让对象更加完善,通过访问权限选择性的将其接⼝提供给外部的用户使用。
2.public修饰的成员在类外可以直接被访问;protected和private修饰的成员在类外不能直接被访问。
3.访问权限作用域从该访问限定符出现的位置开始直到下一个访问限定符出现时为止,如果后面没有访问限定符,作用域就到 }即类结束。
4.class定义成员没有被访问限定符修饰时默认为private,struct默认为public。
5.⼀般成员变量都会被限制为private/protected,需要给别⼈使用的成员函数会放为public。
在这里插入图片描述

3、类域

1.类定义了⼀个新的作用域,类的所有成员都在类的作用域中,在类体外定义成员时,需要使用 :: 作用域操作符指明成员属于哪个类域。
2.类域影响的是编译的查找规则,下面程序中Init如果不指定类域Stack,那么编译器就把Init当成全局函数,那么编译时,找不到array等成员的声明/定义在哪里,就会报错。指定类域Stack,就是知道Init是成员函数,当前域找不到的array等成员,就会到类域中去查找。

#include<iostream>
using namespace std;
class Stack
{ 
public:
	// 成员函数
	void Init(int n = 4);
private:
	// 成员变量
	int* array;
	size_t capacity;
	size_t top;
};
// 声明和定义分离,需要指定类域
void Stack::Init(int n)
{
	array = (int*)malloc(sizeof(int) * n);
	if (nullptr == array)
	{
	perror("malloc申请空间失败");
	return;
	} 
	capacity = n;
	top = 0;
} 
int main()
{
	Stack st;
	st.Init();
	return 0;
}

一、实例化

1、实例化概念

1.用类类型在物理内存中创建对象的过程,称为类实例化出对象。
• 类是对象进行一种抽象描述,是一个模型一样的东西,限定了类有哪些成员变量,这些成员变量只是声明,没有分配空间,用类实例化出对象时,才会分配空间。
2.一个类可以实例化出多个对象,实例化出的对象 占用实际的物理空间,存储类成员变量。类就像设计图一样,不能存储数据,实例化出的对象才能分配物理内存存储数据。

#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
	Date d1;
	Date d2;
	d1.Init(2024, 3, 31);
	d1.Print();
	d2.Init(2024, 7, 5);
	d2.Print();
	return 0;
}

2、对象大小

类实例化出的每个对象,都有独的立数据空间,所以对象中肯定包含成员变量,那么成员函数是否包含呢?
首先先函数被编译后是⼀段指令,对象中没办法存储,这些指令
存储在一个单独的区域(代码段),那么对象中非要存储的话,只能是成员函数的指针。其实函数指针是不需要存储的,函数指针是一个地址,调用函数被编译成汇编指令[call 地址], 其实编译器在编译链接时,就要找到函数的地址,不是在运行时找,只有动态多态是在运行时找,就需要存储函数地址。
对象中只存储成员变量,C++规定类实例化的对象也要符合内存对齐的规则。

3、内存对齐规则

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

三、this指针

• Date类中有 Init 与 Print 两个成员函数,函数体中没有关于不同对象的区分,那当d1调用Init和Print函数时,该函数是如何知道应该访问的是d1对象还是d2对象呢?
那么这里就要看到C++给了一个隐含的this指针解决这里的问题
1.编译器编译后,类的成员函数默认都会在形参第一个位置,增加一个当前类类型的指针,叫做this指针。比如Date类的Init的真实原型为, void Init(Date* const this, int year,int month, int day)
2.类的成员函数中访问成员变量,本质都是通过this指针访问的,如Init函数中给_year赋值, this->_year = year;
• C++规定不能在实参和形参的位置显示的写this指针(编译时编译器会处理),但是可以在函数体内显示使用this指针。

#include<iostream>
using namespace std;
class Date
{ 
public:
	// void Init(Date* const this, int year, int month, int day)
	void Init(int year, int month, int day)
	{
		// 编译报错:error C2106: “=”: 左操作数必须为左值
		// this = nullptr;
		// this->_year = year;
		_year = year;
		this->_month = month;
		this->_day = day;
	} 
	void Print()
	{
		cout << _year << "/" << _month << "/" << _day << endl;
	}
private:
	// 这⾥只是声明,没有开空间
	int _year;
	int _month;
	int _day;
};
int main()
{
	// Date类实例化出对象d1和d2
	Date d1;
	Date d2;
	// d1.Init(&d1, 2024, 3, 31);
	d1.Init(2024, 3, 31);
	d1.Print();
	d2.Init(2024, 7, 5);
	d2.Print();
	return 0;
}

http://www.niftyadmin.cn/n/5556538.html

相关文章

【JVM】JVM实战笔记-随笔

JVM实战笔记-随笔 前言字节码如何查看字节码文件jclasslibJavapArthasArthurs监控面板Arthus查看字节码信息 内存调优内存溢出的常见场景解决内存溢出发现问题Top命令VisualVMArthas使用案例 Prometheus Grafana案例 堆内存情况对比内存泄漏的原因:代码中的内存泄漏并发请求问…

【Linux】Linux操作系统

Linux基本指令 os概念与定位 本节内容&#xff1a; Linux操作系统讲解 os概念与定位 操作系统&#xff08;Operating System&#xff0c;简称OS&#xff09;是管理和控制计算机硬件与软件资源的计算机程序。总的来讲&#xff0c;操作系统是一款做软硬件管理的软件。 了解操作…

IEC104初学者教程,第四章:IEC 104 开发环境搭建

第四章&#xff1a;IEC 104 开发环境搭建 文章目录 第四章&#xff1a;IEC 104 开发环境搭建IEC104从站模拟器的使用IEC104主站模拟器的使用 为了搭建开发环境&#xff0c;我们需要准备三款软件&#xff1a; 主站下载地址&#xff1a;IEC104主站模拟器从站下载地址&#xff1a;…

设计模式使用场景实现示例及优缺点(行为型模式——命令模式)

从前&#xff0c;在一个美丽而神秘的王国里&#xff0c;住着一位智慧而仁慈的国王。他不仅以其公正和睿智著称&#xff0c;还因为他对知识的热爱和追求。他的王国繁荣昌盛&#xff0c;人们生活幸福安康。但即便如此&#xff0c;国王知道&#xff0c;要维持这种繁荣与和平&#…

Leetcode算法题(移除链表中的元素)

题目如下&#xff1a; 思路1&#xff1a;创建一个新的带头链表 &#xff08;newhead&#xff09;&#xff0c;遍历头结点对应的值分别于x进行比较&#xff0c;将不等于x的节点尾插到新的带头链表中&#xff0c;返回新的带头链表的下一个节点。 代码如下&#xff1a; typedef …

OpenCV教程:cv2如何把两张图片的大小,设置成相同的宽高

-------------OpenCV教程集合------------- Python教程99&#xff1a;一起来初识OpenCV&#xff08;一个跨平台的计算机视觉库&#xff09; OpenCV教程01&#xff1a;图像的操作&#xff08;读取显示保存属性获取和修改像素值&#xff09; OpenCV教程02&#xff1a;图像处理…

深度解析:如何优雅地删除GitHub仓库中的特定commit历史

&#x1f49d;&#x1f49d;&#x1f49d;欢迎莅临我的博客&#xff0c;很高兴能够在这里和您见面&#xff01;希望您在这里可以感受到一份轻松愉快的氛围&#xff0c;不仅可以获得有趣的内容和知识&#xff0c;也可以畅所欲言、分享您的想法和见解。 推荐:「stormsha的主页」…

MySQL索引特性(上)

目录 索引的重要 案例 认识磁盘 MySQL与存储 先来研究一下磁盘 扇区 定位扇区 结论 磁盘随机访问与连续访问 MySQL与磁盘交互基本单位 建立共识 索引的理解 建立测试表 插入多条记录 局部性原理 所有的MySQL的操作(增删查改)全部都是在MySQL当中的内存中进行的&am…