【C++】初始化列表、匿名对象、static成员、友元、内部类

news2025/3/3 4:55:00

文章目录

  • 一、初始化列表
    • 构造函数体赋值
    • 初始化列表
    • explicit关键字
  • 二、匿名对象
  • 三、static成员
  • 四、友元
    • 友元函数
    • 友元类
  • 五、内部类
  • 六、练习题

一、初始化列表

构造函数体赋值

实际上,构造函数的函数体内,并不是对 对象 初始化的地方,而是对成员变量进行赋值。因为初始化只能初始化一次,而构造函数体内可以多次赋值。

class Date
{
public:
	Date(int year, int month, int day)
	{
		_year = year;
		_year++;//二次赋值
		_month = month;
		_day = day;
	}
private:
	int _year;
	int _month;
	int _day;
};

真正进行初始化的地方是初始化列表(在创建类变量时,初始化列表将成员变量直接初始化为括号内的表达式值)

初始化列表

初始化列表:以一个冒号开始,接着是一个以逗号分隔的数据成员列表,每个成员变量后面跟一个放在括号中的初始值或表达式。

class Date
{
public:
	Date(int year, int month, int day)
		: _year(year)	//冒号
		, _month(month) //逗号
		, _day(day)		//逗号
	{}
	//也可以写成一行
	Date(int year,int month,int day):_year(year),_month(month),_day(day) {}
	
private:
	int _year;
	int _month;
	int _day;
};

{}内仍可以进行其他操作,比如栈的初始化列表

Stack(int capacity = 4)
	: _a((int*)malloc(sizeof(int) * capacity))
	, _top(0)
	, _capacity(capacity)
{
	//进行其他操作
	if (nullptr == _a)
	{
		perror("malloc");
		return;
	}
}

初始化列表规则

  • 1.每个成员变量在初始化列表中只能出现一次。(因为初始化只能初始化一次)

  • 2.类中包含有:引用成员变量、const成员变量、自定义类型成员(且该类没有默认构造函数时),必须放在初始化列表进行初始化,不能在函数体内通过语句赋值。
    在这里插入图片描述
    对于自定义类型将会调用它的默认构造函数,没有找到默认构造就会报错;这种情况只能在初始化列表初始化。
    引用变量const变量必须在定义时初始化,放在函数体内部就不是初始化而是赋值;所以对于类的引用成员变量const成员变量必须在初始化列表进行初始化。

  • 3.初始化列表的初始化顺序的要与类中的声明顺序一致,与在初始化列表的顺序无关。
    在这里插入图片描述
    类中先声明的_a1,再声明_a2,所以在初始化列表进行初始化时,会先初始化_a1,赋给它_a2的值,但此刻_a2没有初始化,所以为随机值。

虽然构造函数和初始化列表都能完成初始化的工作,但是建议尽量使用初始化列表。因为不管是否使用初始化列表,对于自定义类型成员变量, 一定会先使用初始化列表初始化。并且初始化列表可以提高程序的效率和可读性,因为它可以避免在构造函数体中进行初始化,从而减少了构造函数的执行时间。

explicit关键字

隐式类型转换

首先,在C语言中我们学过,对于内置基本类型如int,char,double等不同类型之间赋值是会发生隐式类型转换的。

int x = 1;
double y = x;
const double& z = x;

int类型的x赋值给double类型的y的过程中,会产生一个临时变量,并且这个临时变量是具有常性的;也就是说,x会先生成一个const double类型的常量,再将这个常量赋值给y。此时y只是x的值的拷贝,y的改变不会影响x。

但是引用是变量的别名,二者共用一块空间。产生的临时变量具有常性,所以必须要用常引用来接收临时变量。

在这里插入图片描述
引用变量类型和实体类型不同,要使用常引用接收。

同样,构造函数也会发生隐式类型转换,但编译器会进行优化。

class A
{
public:
	A(int x = 0)//构造
		: _x(x)
	{
		cout << "A(int x = 0)" << endl;
	}
	A(const A& a)//拷贝构造
		: _x(a._x)
	{
		cout << "A(const A& a)" << endl;
	}
private:
	int _x;
	int _y;
};
int main()
{
	A a1(10);//构造
	a1 = 20;//构造+拷贝构造-->优化为构造
	return 0; 
}

入图片描述](https://img-blog.csdnimg.cn/direct/f2d49ae3ab57435e8935bc416ce59930.png)

用explicit关键字修饰构造函数,会禁止隐式类型转换。

在这里插入图片描述

二、匿名对象

在C++中,匿名对象是指在没有被命名的情况下创建的临时对象。 它们通常用于在单个语句中执行一系列操作或调用某个函数,并且不需要将其结果存储到变量中。

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

在这里插入图片描述

  • 匿名对象生命周期在当前行
  • const引用可以延长匿名对象的生命周期,使其生命周期在当前函数局部域
  • 常规调用成员函数就是先实例化一个类对象,再通过对象去调用成员函数;而匿名对象不用实例化可以直接调用成员函数,虽然方便但是只能调用一次
  • 对于内置类型的匿名对象,其结果默认是0。

在这里插入图片描述

三、static成员

类的静态成员分为两类:

用static修饰的成员变量,称之为静态成员变量
用static修饰的成员函数,称之为静态成员函数

静态成员变量一定要在类外进行初始化

static成员特性

  • 1.静态成员变量必须在类外定义并初始化,定义时不加static关键字,要加域作用限定符::,表明是哪个类域的静态成员变量。
    在这里插入图片描述
  • 2.静态成员函数没有隐藏的this指针,不能访问任何非静态成员。
    在这里插入图片描述
    总结:静态(没有this指针)不能调用非静态(需要this指针),但是非静态可以调用静态
  • 3.静态成员为所有类对象所共享,不属于某个具体的实例。
  • 也就是说静态成员变量不计入类对象所占用的大小空间,因为不属于某一个具体的类对象。
  • 调用静态成员的方式有:类名::静态成员、对象.静态成员、通过匿名对象。

对于公有的静态成员变量,访问方式如下:

class A
{
public:
	static int GetACount()
	{
		return _count;
	}
//private:
	//公有
	static int _count;
};
int A::_count = 0;
int main()
{
	A a;
	cout << A::_count << endl;//指定类域和访问限定符
	cout << a._count << endl;//通过对象
	cout << A()._count << endl;//匿名对象
	return 0;
}

对于私有的静态成员变量,我们只能间接通过静态成员函数来访问:

class A
{
public:
	static int GetACount()
	{
		return _count;
	}
private:
	static int _count;
};
int A::_count = 0;
int main()
{
	A a;
	cout << A::GetACount() << endl;
	cout << a.GetACount() << endl;
	cout << A().GetACount() << endl;
	return 0;
}

4.静态成员和类的普通成员一样,也有public、protected、private3种访问级别,也可以具有返回值

注意:

1.静态成员函数不可以调用非静态成员函数(因为静态成员函数没有this指针)
2.非静态成员函数可以调用类的静态成员函数
3.静态成员变量一定要在类外进行初始化

四、友元

私有成员想让外部类或函数访问,就需要用到友元。

通俗地讲,友元的作用就是邀请朋友(其他类或类外的函数)来自己家中做客(访问自己类的中private私有成员和protect保护成员)

友元关键字friend

友元分为:友元函数友元类

友元函数

友元函数只是在类内声明,在类外定义并不是类的成员函数。友元函数可以在类内任意地方声明,不受访问限定符限制。

1.全局函数做友元

在这里插入图片描述

一个全局函数声明为多个类的友元函数,需要向前引用声明,否则会报错。

class B;//一定要向前引用声明

class A
{
public:
	A(int a = 0) : _a(a) {}
	friend void Print(A& aa, B& bb);//声明为A类的友元函数
private:
	int _a;
};

class B
{
public:
	B(int b = 0) : _b(b) {}
	friend void Print(A& aa, B& bb);//声明为B类的友元函数
private:
	int _b;
};

void Print(A& aa, B& bb)
{
	cout << aa._a << endl;
	cout << bb._b << endl;
}

int main()
{
	A a(10);
	B b(20);
	Print(a , b);
	return 0;
}

因为A类中使用了B类,所以要在A前面先声明B。

2.成员函数做友元

友元函数是其他类的成员函数,也需要向前引用声明
注意:声明在前,定义在后。并且成员函数声明要放在友元函数声明的前面

class A;//向前引用声明

class B
{
public:
	B(int b = 0) : _b(b) {}
	void Print(A& aa);//成员函数声明
private:
	int _b;
};

class A
{
public:
	A(int a = 0) : _a(a) {}
	friend void B::Print(A& aa);//声明为A类的友元函数
private:
	int _a;
};

void B::Print(A& aa)//定义
{
	cout << aa._a << endl;
}

int main()
{
	A a(10);
	B b(20);
	b.Print(a);
	return 0;
}

注意:

友元函数不能用const修饰
友元函数可访问类的私有和保护成员,但不是类的成员函数

有两个特殊的操作符流插入<<流提取>> 分别搭配coutcin使用。对于内置类型,可以直接使用这两个操作符,并且可以自动识别类型。但是对于自定义类型,我们需要自己写这两个的运算符重载。但是这两个是无法重载为类的成员函数的,重载为全局函数又无法访问私有成员,那么就需要用到友元;具体的实现方法放到运算符重载篇一起总结。

友元类

一个类如果是另一个类的友元类,那么这个类的所有成员函数都可以访问另一个类的非公有成员。

class A
{
	friend class B;//B声明为A的友元
public:
	A(int a = 0) : _a(a) {}
private:
	int _a;
};

class B
{
public:
	B(int b = 0) : _b(b) {}

	//直接访问A类的私有成员
	void Func(int x)
	{
		_a1._a = x;
	}
private:
	A _a1;
	int _b;
};

友元类的特性:

1.友元关系是单向的,不具有交换性。
例如:B是A的友元类,在B类中可以直接访问A的非公有成员,但在A类中不能访问B的非公有成员。 >
2.友元关系不能传递
例如:B是A的友元,C是B的友元,不能说明C是A的友元。

五、内部类

如果一个类定义在另一个类的内部,这个类就叫做内部类。

注意: 内部类是一个独立的类,并不属于外部类,不能通过外部类去访问内部类的成员。但内部类是外部类的友元

内部类的特性:

1.内部类可以定义在外部类的任何地方,但是受访问限定符的限制。
内部类如果定义在public,可以通过外部类名::内部类名来定义对象;如果定义在private则不可定义内部类的对象。
2.内部类可以不通过外部类的对象或类名,去直接访问外部类的static成员;但不能直接访问外部类的成员函数。
3.内部类不占外部类的大小空间。也就是说,外部类的大小与内部类无关。

class A
{
public:
	A(int a = 0) : _a(a) {}
	static int GetACount()
	{
		return _count;
	}
private:
	int _a;
	static int _count;

public:
	class B//内部类B是外部类A的友元
	{
	public:
		B(int b = 0) : _b(b) {}

		//直接访问A类的私有成员
		int Print(const A& a)
		{
			cout << a._a << endl;
			_count++;//访问外部类的static成员变量
			return GetACount();//访问外部类的static成员函数
		}
	private:
		int _b;
	};
};

int A::_count = 0;
int main()
{
	A aa(10);
	A::B bb(20);//通过外部类A来创建内部类B的对象
	bb.Print(aa);
	return 0;
}

4.内部类可以在外部类中声明,然后在外面定义。

class A
{
private:
	int _a;
public: class B;//内部类声明
};

class A::B//内部类定义
{
public:
	B(int b = 0) : _b(b) {}
private:
	int _b;
};

六、练习题

描述

求1+2+3+…+n,要求不能使用乘除法、for、while、if、else、switch、case等关键字及条件判断语句(A?B:C)。
数据范围: 0<n≤2000<n≤200
进阶: 空间复杂度 O(1)O(1) ,时间复杂度 O(n)O(n)

题目地址

示例1:

输入:5
返回值:15

示例2:

输入:1
返回值:1

题目的意思就是不让我们用常规的简单做法。非常规的话,最简单省事的方法就是用与运算和递归,与本节内容无关,就不多说了。这里介绍一种方法,需要用到类的性质。

思路

  • 结合前面学习的内容,我们知道,类实例化对象时会自动调用构造函数,那么实例化n个对象就会调用n次构造函数。利用这一特性,我们可在在构造函数中进行累加和,然后实例化一个n个大小的数组即可。
  • 要保证每次调用构造函数时变量的值能继承上一次的结果,我们可以用静态成员变量来存储。
  • 而这道题是核心代码模式,结果是由系统给的类的成员函数来返回,只能在成员函数中实例化对象,那么我们就可以用内部类。
class Solution
{
    class Sum//内部类
    {
    public:
        Sum()//构造
        {
        	//内部类可以直接访问外部类的static成员
            _i++;
            _ret += _i; 
        }
    };
public:
    int Sum_Solution(int n)
    {
        Sum a[n];//实例化n个Sum类对象,调用n次构造
        return _ret;
    }
private:
    static int _i;
    static int _ret;
};

int Solution::_i = 0;
int Solution::_ret = 0;

至此,类与对象的内容总结完毕。我们要理解:类是对某一类实体(对象)来进行描述的,描述该对象具有哪些属性(成员变量),哪些方法(成员函数),描述完成后就形成了一种新的自定义类型,用该自定义类型就可以实例化具体的对象。

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

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

相关文章

电子杂志制作的必备软件:轻松提升制作效率

​电子杂志作为一种新型的媒体形式&#xff0c;具有互动性强、内容丰富、传播范围广等特点。随着互联网的普及&#xff0c;越来越多的企业和个人开始关注和投入电子杂志的制作。然而&#xff0c;电子杂志的制作过程往往复杂繁琐&#xff0c;需要付出大量的时间和精力。为了提高…

微信小程序学习(六):常用原生 API

&#x1f517;API官方文档 1、网络请求 wx.request({// 接口地址&#xff0c;仅为示例&#xff0c;并非真实的接口地址url: example.php,// 请求的参数data: { x: },// 请求方式 GET|POST|PUT|DELETEmethod: GET,success (res) {console.log(res.data)},fail(err) {console.…

本地离线模型搭建指南-本地运行显卡选择

搭建一个本地中文大语言模型&#xff08;LLM&#xff09;涉及多个关键步骤&#xff0c;从选择模型底座&#xff0c;到运行机器和框架&#xff0c;再到具体的架构实现和训练方式。以下是一个详细的指南&#xff0c;帮助你从零开始构建和运行一个中文大语言模型。 本地离线模型搭…

05 - matlab m_map地学绘图工具基础函数 - 设置比例尺指北针

05 - matlab m_map地学绘图工具基础函数 - 设置比例尺指北针 0. 引言1. 关于m_scale2. 关于m_ruler3. 关于m_northarrow4. 结语 0. 引言 本篇介绍下m_map中添加指北针(m_northarrow)、比例尺(m_ruler)和进行比例缩放(m_scale)的函数及其用法 。 1. 关于m_scale m_scale用于图件…

LeetCode 算法:二叉树的中序遍历 c++

原题链接&#x1f517;&#xff1a;二叉树的中序遍历 难度&#xff1a;简单⭐️ 题目 给定一个二叉树的根节点 root &#xff0c;返回 它的 中序 遍历 。 示例 1&#xff1a; 输入&#xff1a;root [1,null,2,3] 输出&#xff1a;[1,3,2] 示例 2&#xff1a; 输入&…

【Pandas驯化-14】一文搞懂Pandas中的时间处理函数date_range、resample、shift技巧

【Pandas驯化-14】一文搞懂Pandas中的时间处理函数date_range、resample、shift技巧 本次修炼方法请往下查看 &#x1f308; 欢迎莅临我的个人主页 &#x1f448;这里是我工作、学习、实践 IT领域、真诚分享 踩坑集合&#xff0c;智慧小天地&#xff01; &#x1f387; 相关…

PHP米表域名出售管理源码带后台

源码介绍 html5米表源码PHP域名销售程序安装方法&#xff1a; 本站已测试,各项功能正常,功能易用,不复杂,非常适合个人米表使用 1、所有文件传至网站目录 2、浏览器执行http://你的访问网址/install 3、输入mysql帐号及密码信息&#xff0c;提交安装 源码截图 源码下载 …

华为od-C卷200分题目3 - 两个字符串间的最短路径问题

华为od-C卷200分题目3 - 两个字符串间的最短路径问题 题目描述 给定两个字符串&#xff0c;分别为字符串A与字符串B。 例如A字符串为ABCABBA&#xff0c;B字符串为CBABAC可以得到下图m*n的二维数组&#xff0c;定义原点为(0, 0)&#xff0c;终点为(m, n)&#xff0c;水平与垂…

【Android逆向】小白也能学会的一个小时破解某猫社区VIP会员

第二步&#xff1a;使用 dex2jar 将 classes.dex 转成 jar 文件 cmd到dex2jar文件夹目录&#xff0c;执行 d2j-dex2jar D://xxx/xxx/classes.dex 得到 jar 文件 静态分析 拿到源码后&#xff0c;首先我们需要找到应用的限制点&#xff0c;绕过App里面的判断。 然后分析源码&…

520. 检测大写字母

题目 我们定义&#xff0c;在以下情况时&#xff0c;单词的大写用法是正确的&#xff1a; 全部字母都是大写&#xff0c;比如 “USA” 。单词中所有字母都不是大写&#xff0c;比如 “leetcode” 。如果单词不只含有一个字母&#xff0c;只有首字母大写&#xff0c;比如 “Go…

React学习(二)——状态(数据)与状态修改

useState 在React中&#xff0c;useState 是一个非常重要的Hook&#xff0c;它允许你在函数组件中添加“状态”&#xff08;state&#xff09;。在传统的React类组件中&#xff0c;我们使用this.state来管理和更新组件的状态。然而&#xff0c;在函数组件中&#xff0c;由于它们…

【Docker】Docker下载安装_使用阿里云加速配置

1、下载安装 1.1前提条件 安装环境&#xff1a; 目前&#xff0c;CentOS 仅发行版本中的内核支持 Docker。Docker 运行在 CentOS 7 上&#xff0c;要求系统为64位、系统内核版本为 3.10 以上。Docker 运行在 CentOS-6.5 或更高的版本的 CentOS 上&#xff0c;要求系统为64位…

管理不到位,活该执行力差?狠抓这4点要素,强化执行力

管理不到位&#xff0c;活该执行力差&#xff1f;狠抓这4点要素&#xff0c;强化执行力 一&#xff1a;强化制度管理 1、权责分明&#xff0c;追责管理 要知道&#xff0c;规章制度其实就是一种“契约”。 在制定制度和规则的时候&#xff0c;民主一点&#xff0c;征求团队成员…

基于格网的边缘点检测(python)

1、背景介绍 前文已介绍对点云进行格网处理&#xff0c;可以计算平面点云面积、格网拓扑关系构建&#xff0c;相关博客如下&#xff1a; &#xff08;1&#xff09;点云格网过程可视化&#xff08;C PCL&#xff09;-CSDN博客 &#xff08;2&#xff09;平面点云格网过程及可…

一小时搞定Git(含盖IDEA使用)

文章目录 1. git基本概念1.1版本控制1.1.1 版本控制软件 2. 命令的使用2.1 Linux命令2.2 git基础指令2.2.1 设置用户2.2.2 初始化本地仓库2.2.3 查看本地仓库状态2.2.4 添加暂存区域2.2.5 提交本地库2.2.6 切换版本 2.3 分支操作2.3.1 分支基本操作2.3.2 合并操作2.3.4 分支开发…

C语言| 数组的插入

在下标为index的位置插入一个数字 1 定义数组a&#xff0c;数组b存放插入元素后的数组&#xff0c;下标index 值num 循环变量i 2 输入要插入的位置下标和数值 3 for循环 嵌套if多层语句 if数组的最大下标i < index,说明插入元素的位置在数组中不存在&#xff0c;系统随机分…

Python使用tkinter制作无边框透明时钟源码讲解(tkinter如何实现窗口无边框透明)

文章目录 📖 介绍 📖🏡 演示环境 🏡📒 文章内容 📒📝 导入必要的库📝 创建主窗口🎯 去掉窗口边框🎯 设置窗口透明度🎯 允许窗口背景透明🎯 设置窗口背景颜色为透明🎯 设置窗口位置🎯 创建用于显示时间的标签📝 更新时间函数📝 使窗口可移动📝…

winRar去广告记录

效果&#xff1a;双击winRar.exe&#xff0c;不会弹出广告窗口&#xff0c;但会弹出使用时间许可警告&#xff0c;效果不是很完美。 工具&#xff1a;everything.exe&#xff08;非必须&#xff09;、sublime text&#xff08;非必须&#xff09;、spyxx.exe&#xff08;非必须…

python-今年第几天

[题目描述] 定义一个结构体变量&#xff08;包括年、月、日&#xff09;。 计算该日在本年中是第几天&#xff0c;注意闰年问题。输入格式&#xff1a; 年 月 日。输出格式&#xff1a; 当年第几天。样例输入 2000 12 31样例输出 366 数据范围 对于100%的数据&#xff0c;保…

啥移动硬盘格式能更好兼容Windows和Mac系统 NTFS格式苹果电脑不能修改 paragon ntfs for mac激活码

对于同时使用Windows和Mac操作系统的用户而言&#xff0c;选择一个既能确保数据互通又能满足大容量存储需求的移动硬盘格式尤为重要。下面我们来看看啥移动硬盘格式能更好兼容Windows和Mac系统&#xff0c;NTFS格式苹果电脑不能修改的相关内容。 一、啥移动硬盘格式能更好兼容…