【C++】类和对象(四)

news2025/1/10 11:19:33

前言:在类和对象中,我们走过了十分漫长的道路,今天我们将进一步学习类和对象,类和对象这块荆棘地很长,各位一起加油呀。

💖 博主CSDN主页:卫卫卫的个人主页 💞
👉 专栏分类:高质量C++学习 👈
💯代码仓库:卫卫周大胖的学习日记💫
💪关注博主和博主一起学习!一起努力!
在这里插入图片描述


目录标题

  • 类和对象
    • const修饰成员函数
    • 再谈构造函数之函数体赋值
    • explict关键字
    • static成员


类和对象

const修饰成员函数

在C++中,可以使用const关键字来修饰成员函数。const修饰符可以被应用于类的成员函数,这表示该函数不会修改任何类的成员变量。

当一个成员函数被const修饰时,它被称为常量成员函数。常量成员函数承诺不会修改任何成员变量。常量成员函数的定义和声明都必须包含const关键字。
语法格式

返回类型 类名::函数名()const
{
	//函数体
	...
	...
}

下面是一个示例

class MyClass 
{
public:
    void nonConstFunction(int year,int month ,int day); 
    // 非常量成员函数             
    void nonConstFunction() const;  // 常量成员函数
private:
    int _day;
    int _month;
    int _year;
};

void MyClass::nonConstFunction(int year = 1,int month = 1 ,int day =1) 
{
    // 对成员变量进行修改
    _year = year;
    _month = month;
    _day = day;
}

void MyClass::nonConstFunction() const 
{
    // 不对成员变量进行修改
    cout << "year: " << _year << " month: " << _month << " day: " << _day << endl;
}


在上面的示例中,nonConstFunction是一个非常量成员函数,可以修改成员变量。而nonConstFunction的const版本是一个常量成员函数,不会修改成员变量。

那我们思考一下为什么在上面这个示例中,const修饰了以后就无法对里面的成员变量进行修改了?请看下图:
在这里插入图片描述

在上图中我们可以知道,编译器会对this指针加上const进行修饰,让外部变量无法对其修改

如果在上面式子中我们对const修饰的成员函数中的成员变量进行修改,就会出现如下情况,如下所示:
在这里插入图片描述
使用const修饰常量成员函数的好处是可以在常量对象上调用该函数,而不会导致编译错误。这样可以提高代码的可读性和安全性。


注意事项:
C++中const成员函数有以下限制:

  1. const成员函数不能修改类的非静态数据成员。这是因为const成员函数保证不会修改对象的状态,所以不能修改任何非静态数据成员。

  2. const成员函数只能调用其他const成员函数。这是因为const成员函数保证不会修改对象的状态,所以只能调用其他也不会修改对象状态的const成员函数。

  3. const成员函数不能通过指针或引用返回非const指针或引用。这是因为const成员函数要保证不会修改对象的状态,所以不能返回非const指针或引用,否则调用者就可以通过这个指针或引用修改对象的状态。

  4. const成员函数可以被非const对象和const对象调用。非const对象调用const成员函数时会被自动转换为const对象。

  5. const成员函数不能被声明为虚函数。虚函数是根据对象的动态类型来调用的,而const成员函数是根据对象的静态类型来调用的,所以不能将const成员函数声明为虚函数。


再谈构造函数之函数体赋值

在创建对象时,编译器通过调用构造函数,给对象中各个成员变量一个合适的初始值。

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

虽然上述构造函数调用之后,对象中已经有了一个初始值,但是不能将其称为对对象中成员变量的初始化,构造函数体中的语句只能将其称为赋初值,而不能称作初始化。因为初始化只能初始化一次,而构造函数体内可以多次赋值

  • . 初始化列表:构造函数可以通过初始化列表来初始化成员变量。在构造函数的参数列表后面使用冒号(:)来定义初始化列表,然后通过成员变量名称和初始值来初始化成员变量(当然这个括号里面也可以写一个有返回值的表达式)。例如:
    在这里插入图片描述
    在这里插入图片描述

注意:

  1. 每个成员变量在初始化列表中只能出现一次(初始化只能初始化一次)
  2. 类中包含以下成员,必须放在初始化列表位置进行初始化:
    引用成员变量,const成员变量,自定义类型成员(该类没有默认构造函数)

实例演示

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

class d
{
public:
	d(int a, int b, int c)
		:_aa(a),
		_b(b),
		_c(c)
	{
		cout << "_b: " << _b << endl;
		cout << "_c: " << _c << endl;
	}

private:
	A _aa;//自定义变量
	int& _b;//引用
	const int _c;//const
};

int main()
{
	d d1(10, 20, 30);

	return 0;
}

在这里插入图片描述

  1. 尽量使用初始化列表初始化,因为不管你是否使用初始化列表,对于自定义类型成员变量,一定会先使用初始化列表初始化。
  2. 成员变量在类中声明次序就是其在初始化列表中的初始化顺序,与其在初始化列表中的先后次序无关

在C++中,构造函数体赋值的注意事项如下:

  1. 初始化列表的优先级高于构造函数体内的赋值:如果同时在初始化列表和构造函数体内对成员变量进行赋值,初始化列表中的赋值操作会先于构造函数体内的赋值操作执行。

  2. 成员变量的声明顺序决定了初始化的顺序:成员变量在类中的声明顺序决定了它们初始化的顺序。在构造函数体内的赋值操作也会按照成员变量的声明顺序进行。

  3. 成员变量的初始化顺序与初始化列表中的顺序一致:在初始化列表中的顺序决定了成员变量初始化的顺序。如果在初始化列表中没有给出某个成员变量的初始值,该成员变量会按照默认构造函数进行初始化。

  4. 常量成员变量必须在初始化列表中进行初始化:对于类中的常量成员变量(const类型),它们必须在初始化列表中进行初始化,而不能在构造函数体内进行赋值操作。

  5. 使用成员初始化列表可以提高效率:使用成员初始化列表可以在对象构造时直接进行赋值操作,避免了先构造默认对象再赋值的额外开销,因此可以提高效率。

总体而言,通过初始化列表进行成员变量的赋值是更好的选择,除非有特殊需要,如需要在构造函数体内做其他逻辑处理,才使用构造函数体内的赋值语句。


explict关键字

在C++中,explicit也是一个关键字,但其使用情况与C#中的略有不同。

在C++中,explicit关键字可以用于单参数构造函数(或转换函数),以防止编译器进行隐式类型转换。默认情况下,单参数构造函数可以用于隐式类型转换。但是,当我们使用explicit关键字来修饰该构造函数时,它将变为只能进行显式类型转换的构造函数,禁止隐式转换。

下面是一个使用explicit关键字的示例:

class Number {
private:
    int value;
public:
    explicit Number(int value) : value(value) {}
 
    int getValue() const {
        return value;
    }
};

void printNumber(const Number& number) {
    std::cout << number.getValue() << std::endl;
}

int main() {
    int intValue = 10;

    // 隐式类型转换不被允许
    // Number number = intValue;

    // 显式类型转换
    Number number = Number(intValue);

    printNumber(number);

    return 0;
}

在这里插入图片描述

在上面的示例中,Number类定义了一个带有单参数的构造函数,并使用explicit关键字进行修饰。在main函数中,我们首先声明一个int类型的变量intValue,并将其赋值为10。然后,我们尝试使用隐式类型转换将intValue转换为Number类型,但由于Number类的构造函数使用了explicit关键字,这会导致编译错误。接着,我们使用显式类型转换将intValue转换为Number类型,并将结果赋值给number变量。最后,我们通过调用printNumber函数,将number对象传递给它并输出结果。可以看到,通过使用explicit关键字,我们可以明确指定我们希望进行显式类型转换,从而避免了隐式类型转换可能带来的意外行为。


static成员

在C++中,static关键字可以应用于类的成员,用于指示该成员是静态的。静态成员与类的实例无关,它们属于整个类而不是类的实例。以下是一些关于C++中静态成员的信息:

  1. 静态数据成员:静态数据成员与类的所有实例共享,它们只有一个副本。可以在类的内部声明并在类的外部初始化静态数据成员必须在类的定义之外进行初始化,并且必须在类的外部定义。并且静态成员变量是在初始化时分配内存的,程序结束时释放内存。
class MyClass 
{
public:
	static int staticData; // 声明静态数据成员
	int unstaticData = 10; //声明并初始化非静态成员
};

int MyClass::staticData = 0; // 初始化静态数据成员

int main() 
{
	MyClass obj1;
	MyClass obj2;
	obj1.staticData = 5;
	cout <<"非静态成员: " << obj1.unstaticData << endl; //输出非静态成员
	cout <<"静态成员: " << obj2.staticData << endl; // 输出静态成员: 5
	return 0;
}

在这里插入图片描述

  1. 静态成员函数:静态成员函数没有访问类的任何实例成员的权限,它们只能访问静态成员。静态成员函数可以通过类名或对象名来引用,非静态成员函数只能通过对象名引用。
class MyClass 
{
public:
    static void StaticFunction()//静态成员函数 
    {
        cout << "Static Function" << endl;
    }
    void UnstaticFunction()//非静态成员函数
    {
        cout << "UnStatic Function" << endl;

    }
};

int main() {
    MyClass d1;
    MyClass::StaticFunction(); // 输出: Static Function
    d1.UnstaticFunction();  //输出: UnStatic Function
    return 0;
}

在这里插入图片描述
静态成员提供了一种在类的所有实例之间共享和访问数据的方式。在某些情况下,静态成员函数可以用作工具函数或全局函数的替代品。然而,静态成员应谨慎使用,因为它们破坏了封装性和面向对象设计的一些原则。


好啦,今天的内容就到这里啦,下期内容预告类和对象(五)友元、内部类、匿名对象等,下期就会对类和对象进行最后的收尾了,各位加油呐!


结语:今天的内容就到这里吧,谢谢各位的观看,如果有讲的不好的地方也请各位多多指出,作者每一条评论都会读的,谢谢各位。


🌏🗺️ 这里祝各位新年快乐 💞💞

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

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

相关文章

《小强升职记:时间管理故事书》阅读笔记

目录 前言 一、你的时间都去哪儿了 1.1 你真的很忙吗 1.2 如何记录和分析时间日志 1.3 如何找到自己的价值观 二、无压工作法 2.1 传说中的“四象限法则 2.2 衣柜整理法 三、行动时遇到问题怎么办&#xff1f; 3.1 臣服与拖延 3.2 如何做到要事第一&#xff1f; 3.…

【碎片知识点】安装Linux系统 VMware与kali

天命&#xff1a;VMware就是可以运行操作系统的载体&#xff0c;kali就是Linux的其中一个分支 天命&#xff1a;Linux有两个分支版本&#xff1a;centos与ubuntu&#xff0c;kali底层就是ubuntu&#xff08;所有Linux用起来都差不多&#xff0c;没啥区别&#xff09; 天命&…

linux系统zabbix自动发现主机

自动发现主机 新的主机浏览器配置创建发现规则创建发现主机后动作 新的主机 rpm -Uvh https://repo.zabbix.com/zabbix/5.0/rhel/7/x86_64/zabbix-release-5.0-1.el7.noarch.rpm# yum clean allyum install zabbix-agentvim /etc/zabbix/zabbix_agentd.conf Server10.12.153.1…

C语言:内存分配---栈区、堆区、全局区、常量区和代码区

一、C语言内存分区 C语言内存分区示意图如下&#xff1a; 1. 栈区 栈区介绍 栈区由编译器自动分配释放&#xff0c;由操作系统自动管理&#xff0c;无须手动管理。栈区上的内容只在函数范围内存在&#xff0c;当函数运行结束&#xff0c;这些内容也会自动被销毁。栈区按内存…

微服务中台架构的设计与实现

本文将探讨微服务中台架构的设计与实现&#xff0c;介绍如何通过微服务的方式进行系统拆分和组合&#xff0c;构建灵活、可扩展且易于维护的中台架构&#xff0c;以加速企业的数字化转型和提升竞争力。 ## 1. 引言 随着企业规模的不断扩大和业务的日益复杂化&#xff0c;传统…

山西电力市场日前价格预测【2024-02-12】

日前价格预测 预测说明&#xff1a; 如上图所示&#xff0c;预测明日&#xff08;2024-02-12&#xff09;山西电力市场全天平均日前电价为127.42元/MWh。其中&#xff0c;最高日前电价为369.24元/MWh&#xff0c;预计出现在18:45。最低日前电价为0.00元/MWh&#xff0c;预计出…

库函数strlen的实现

目录 一、原理二、思路三、实现 一、原理 库函数strlen的功能是求字符串长度&#xff0c;统计的是字符串中 \0 之前的字符的个数。 函数原型如下&#xff1a; size_t strlen ( const char * str );二、思路 参数str接收⼀个字符串的起始地址&#xff0c;然后开始统计字符串中…

WEB APIs(1)

变量声明const&#xff08;修饰常量&#xff09; const优先&#xff0c;如react&#xff0c;基本const&#xff0c; 对于引用数据类型&#xff0c;可用const声明&#xff0c;因为储存的是地址 何为APIs 可以使用js操作HTML和浏览器 分类&#xff1a;DOM&#xff08;文档对象…

C# Winform .net6自绘的圆形进度条

using System; using System.Drawing; using System.Drawing.Drawing2D; using System.Windows.Forms;namespace Net6_GeneralUiWinFrm {public class CircularProgressBar : Control{private int progress 0;private int borderWidth 20; // 增加的边框宽度public int Progr…

全闭环直播推流桌面分享远控系统

直播推流涉及多协议&#xff0c;多端技术栈和知识点&#xff0c;&#xff0c;想要做好并不容易&#xff0c;经过几年时间的迭代&#xff0c;终于小有成就&#xff0c;聚集了媒体服务器&#xff0c;实时会议sfu&#xff0c;远控kvm等功能。可以做一个音视频应用的瑞士小军刀。主…

为什么说技术进步很慢? —— 技术的先进性与其当下价值的不匹配

一、背景 技术进步是否缓慢是一个相对的概念&#xff0c;需要在不同的领域和时间段内进行分析。以下是一些不同领域中可能造成技术进步看似缓慢的原因&#xff1a; 1. **基础研究瓶颈**&#xff1a;许多先进技术的发展依赖于基础科学的突破&#xff0c;而这些突破往往需要长时…

openGauss学习笔记-219 openGauss性能调优-确定性能调优范围-硬件瓶颈点分析-网络

文章目录 openGauss学习笔记-219 openGauss性能调优-确定性能调优范围-硬件瓶颈点分析-网络219.1 查看网络状况 openGauss学习笔记-219 openGauss性能调优-确定性能调优范围-硬件瓶颈点分析-网络 获取openGauss节点的CPU、内存、I/O和网络资源使用情况&#xff0c;确认这些资源…

论文阅读-面向机器学习的云工作负载预测模型的性能分析

论文名称&#xff1a;Performance Analysis of Machine Learning Centered Workload Prediction Models for Cloud 摘要 由于异构服务类型和动态工作负载的高变异性和维度&#xff0c;资源使用的精确估计是一个复杂而具有挑战性的问题。在过去几年中&#xff0c;资源使用和流…

【机器学习笔记】3 逻辑回归

分类问题 分类问题监督学习最主要的类型&#xff0c;主要特征是标签离散&#xff0c;逻辑回归是解决分类问题的常见算法&#xff0c;输入变量可以是离散的也可以是连续的 二分类 先从用蓝色圆形数据定义为类型1&#xff0c;其余数据为类型2&#xff1b;只需要分类1次&#x…

ElasticSearch分词器和相关性详解

目录 ES分词器详解 基本概念 分词发生时期 分词器的组成 切词器&#xff1a;Tokenizer 词项过滤器&#xff1a;Token Filter 停用词 同义词 字符过滤器&#xff1a;Character Filter HTML 标签过滤器&#xff1a;HTML Strip Character Filter 字符映射过滤器&#x…

【Linux技术宝典】深入理解Linux基本指令:命令行新手指南

&#x1f4f7; 江池俊&#xff1a; 个人主页 &#x1f525;个人专栏&#xff1a; ✅数据结构冒险记 ✅Linux技术宝典 &#x1f305; 有航道的人&#xff0c;再渺小也不会迷途。 文章目录 一、Linux下基本指令1. ls 指令2. pwd指令3. clear指令4. cd指令什么是家目录&#xf…

2023全球云计算市场份额排名

关注卢松松&#xff0c;会经常给你分享一些我的经验和观点。 最近Synergy研究院发布了最新的全球云计算市场份额排名。 亚马逊依旧是以31%的的市场份额排名第一&#xff0c;微软azure24%排名第二&#xff0c;Google云11%排名第三&#xff0c;阿里云4%排名第四。腾讯云和IBM、…

thinkphp5.0提示不支持redis,not support: redis

安装PHP扩展 例如宝塔&#xff0c;其他环境请用命令行&#xff0c;安装 redis配置完成以后&#xff0c;修改php.ini把redis扩展打开即可&#xff0c;重启环境

docker磁盘不足!已解决~

目录 &#x1f35f;1.查看docker镜像目录 &#x1f9c2;2.停止docker服务 &#x1f953;3.创建新的目录 &#x1f32d;4.迁移目录 &#x1f37f;5.编辑迁移的目录 &#x1f95e;6.重新加载docker &#x1f354;7.检擦docker新目录 &#x1f373;8.删掉旧目录 1.查看doc…

MySQL 基础知识(五)之数据增删改

目录 1 插入数据 2 删除数据 3 更改数据 创建 goods 表 drop table if exists goods; create table goods ( id int(10) primary key auto_increment, name varchar(14) unique, stockdate date )charsetutf8; 1 插入数据 当要插入的数据为日期/时间类型时&#xff0c;如果…