C++面向对象(下)

news2024/12/26 11:28:49

文章目录

  • 前言
  • 1.再谈构造函数
    • 1.初始化列表
    • 2.explicit关键字
  • 2. static成员
    • 1.概念
  • 3.友元
    • 1.概念
    • 2.友元函数
    • 3.友元类
  • 4. 内部类
  • 5.匿名对象
  • 6.编译器优化
  • 7.总结

前言

本文是主要是将之前关于C++面向对象中的一些没有归纳到的零星知识点进行补充,同时对C++中的面向对象简单收个尾。


1.再谈构造函数

1.初始化列表

在创建对象时,编译器通过调用构造函数,给对象中各个成员变量一个合适的初始值。虽然构造函数调用之后,对象中已经有了一个初始值,但是不能将其称为对对象中成员变量的初始化,构造函数体中的语句只能将其称为赋初值,而不能称作初始化。因为初始化只能初始化一次,而构造函数体内可以多次赋值。

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

代码示例
在这里插入图片描述

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

代码示例

class A
{ //没有默认构造函数
  public:A(int a)
	{
		_a = a;
    }
private:
	 int _a;
};

class B
{ public:
	B(int a=1,int b=1)
	:x(a)
	,y(b)
	,z(1) 
	{
 
	}
private:
	int& x;
	A y;
	const int z;
};

对于B类来说它的每个成员变量都需要在初始化列表中进行初始化。

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

代码示例

#include<iostream>
using namespace std;
class A
{
public:
    A(int a)
        :_a1(a)
        , _a2(_a1)
    {}

    void Print() {
        cout << _a1 << " " << _a2 << endl;
    }
private:
    int _a2;
    int _a1;
};
int main()
{
    A a(1);
    a.Print();
}

打印结果如下

在这里插入图片描述

之所以一个是1另一个是随机值,其实是因为成员变量在类中声明次序就是其在初始化列表中的初始化顺序。先初始化的是_a2但是_a2是用_a1初始化的,这个时候_a1还没有被初始化是随机值,所以_a2就是随机值了。

对于初始化列表的理性看待

在class类中成员变量可以被看作声明,初始化列表相当于给了成员变量一个地方进行初始化定义。正来说一个变量 int a=12这就是对变量初始化且变量被定义出来了。类中的成员变量相当于声明和初始化分离,分别在不同的地方进行声明 初始化。

2.explicit关键字

构造函数不仅可以构造与初始化对象,对于单个参数或者除第一个参数无默认值其余均有缺省值值的构造函数,还具有类型转换的作用。(隐式类型转换)

在这里插入图片描述


当然对于多参数也可以的

在这里插入图片描述

这种初始化方式有点像C语言中结构体的初始化方式,用一个整形变量给对象赋值,实际编译器背后会用构造一个无名对象,最后用无名对象给a对象进行拷贝构造初始化。用explicit修饰构造函数,将会禁止构造函数的隐式转换.

在这里插入图片描述


2. static成员

1.概念

声明为static的类成员称为类的静态成员,用static修饰的成员变量,称之为静态成员变量;用static修饰的成员函数,称之为静态成员函数。静态成员变量一定要在类外进行初始化。

1. 静态成员为所有类对象所共享,不属于某个具体的对象,存放在静态区
2. 静态成员变量必须在类外定义,定义时不添加static关键字,类中只是声明
3. 类静态成员即可用 类名::静态成员 或者 对象.静态成员 来访问
4. 静态成员函数没有隐藏的this指针,不能访问任何非静态成员
5. 静态成员也是类的成员,受public、protected、private 访问限定符的限制

#include<iostream>
using namespace std;
class A
{ public:
    A()
    {
        num++;
    }
    static int get_num()
    {
        return num;
    } 
private:
    static int num;
};
int A::num = 0;
int main()
{
    A a1;
    A a2;
    cout << A::get_num() << endl;
    cout << a1.get_num() << endl;
}

代码中a.get_num和A::都是为了让静态函数突破类域的限制而已,这点和普通成员函数是一样的。static修饰的变量只能被初始化一次这点要注意1. 静态成员函数可以调用非静态成员吗?2. 非静态成员函数可以调用类的静态成员吗?对于问题1来说:静态成员函数不存在this指针所以不能访问类中的成员;对于问题2来说:静态成员是属于类的也就是属于每个对象的,相当于共享的。所以非静态函数可以访问类中的静态成员。


对于某些场景这个静态成员还是比较有用,我们来看一道例题。
在这里插入图片描述

示例题目链接

这道题其实就一般给出的题解是用到了位运算,除此以外还可以用到静态的成员来处理问题。

代码示例

class Sum
{
public:
 Sum()
  {
    _sum+=_i;
    ++_i;
  }
static int sum()
{
    return _sum;
}
  private:
  static int _i;
  static int _sum;
};
 int Sum::_i=1;
 int Sum::_sum=0;

class Solution {
public:
    int Sum_Solution(int n) {
        
        Sum a[n];
        return Sum::sum();
    }
};

这就利用了static函数是属于所有类对象的特性,每调用构造函数一次sum就会+=i一次,创建n个对象就会+= n次这就实现了1到n的求和


3.友元

1.概念

友元提供了一种突破封装的方式,有时提供了便利。但是友元会增加耦合度,破坏了封装,所以友元不宜多用。友元分为:友元函数和友元类。

2.友元函数

友元破坏了封装,为啥还要有友元这种东西呢?现在这样一个场景:尝试去对某个类重载operator<<,然后发现没办法将operator<<重载成成员函数。因为隐含的this指针在抢占第一个参数的位置。this指针默认是第一个参数也就是左操作数了。但是实际使用中cout需要是作为第一个形参对象,才能比较符合正常逻辑。所以要将operator<<重载成全局函数。但又会导致类外没办法访问成员,此时就需要友元来解决。

代码示例

#include<iostream>
using namespace std;
class Date
{ public:
    friend ostream& operator<<( ostream& out, const Date& d);
    Date(int year = 1, int month = 1, int day = 1)
    {
        _year = year;
        _month = month;
        _day = day;
    }
private:
    int _year;
    int _month;
    int _day;
};

ostream& operator<<( ostream& out, const Date& d)
{
    out << d._year << " " << d._month << " " << d._day << endl;
    return out;
}

int main()
{
    Date d1(2022, 1, 1);
    cout << d1;
}

cout其实是一个ostream类型的对象,cout之所以能自动识别类型就是用到了运算符重载和函数重载。对于内置类型,C++官网库中写了内置类型的<<运算符重载,同时又由于函数重载opertor<<,这样使得cout能够处理不同的内置类型。但是对于自定义类来说如果想使用cout要么修改官方库,要么就是自己手动实现operator<<。对于前者,实现肯定是不现实的,只能采用后者方式。如果采用后者肯定不能将这个函数写到内里面,不然第一个操作数就是类对象了这样使用方式就会很别扭,为了保持cout原有的使用方式,只能将这个operator<<写到类内外,但是又需要用访问类中成员,所以只能使用友元函数了。同时返回值传引用是为了连续的输出打印显示保持运算符的特性。

友元函数可以直接访问类的私有成员,它是定义在类外部的普通函数,不属于任何类,但需要在类的内部声明,声明时需要加friend关键字。


3.友元类

友元类的所有成员函数都可以是另一个类的友元函数,都可以访问另一个类中的非公有成员。友元关系是单向的,不具有交换性。比如Time类和Date类,在Time类中声明Date类为其友元类,那么可以在Date类中直接访问Time类的私有成员变量,但想在Time类中访问Date类中私有的成员变量则不行。友元关系不能传递如果C是B的友元, B是A的友元,则不能说明C时A的友元。友元关系不能继承。

代码示例

class Time
{
    friend class Date;   // 声明日期类为时间类的友元类,则在日期类中就直接访问Time类
    public:
        Time(int hour = 0, int minute = 0)
        {
            ;
        }

private:
    int _hour;
    int _minute;
};
class Date
{
public:
    Date(int year = 1000, int month = 1, int day = 1)
    {
        ;
    }

    void SetTimeOfDate(int hour, int minute, int second)
    {
        // 直接访问时间类私有的成员变量
        _t._hour = hour;
        _t._minute = minute;
    }

private:
    int _year;
    int _month;
    int _day;
    
        Time _t;
};

4. 内部类

概念:如果一个类定义在另一个类的内部,这个内部类就叫做内部类。内部类是一个独立的类,它不属于外部类,更不能通过外部类的对象去访问内部类的成员。外部类对内部类没有任何优越的访问权限。注意:内部类就是外部类的友元类,参见友元类的定义,内部类可以通过外部类的对象参数来访问外部类中的所有成员。但是外部类不是内部类的友元。

特性:

1. 内部类可以定义在外部类的public、protected、private都是可以的。
2. 注意内部类可以直接访问外部类中的static成员,不需要外部类的对象/类名。
3. sizeof(外部类)=外部类,和内部类没有任何关系。


5.匿名对象

匿名对象有点想匿名结构体,表示方法是:类名()。匿名对象的生命周期极短只存在于它所在的那一行处。

代码示例

#include<iostream>
using namespace std;
class A
{
public:
	A(int a=1, int b=1, int c=1)
	{
		_a = a;
		_b = b;
		_c = c;
	}
private:
	int _a;
	int _b;
	int _c;
};

int main()
{  
	//匿名对象
	A(2022, 2, 2);
	
}

匿名对象将相当于用一次性杯子,用过就仍。它实用于某些一次性场景。可以作为某些函数的返回值优化性能。


6.编译器优化

在传参和传返回值的过程中,一般编译器会做一些优化,减少对象的拷贝,这个在一些场景下还是非常有用的。

代码示例

#include<iostream>
using namespace std;
class A
{
public:
	A(int a=1)
	{
		_a = a;
		cout << "直接构造" << endl;
	}
	A(const A& a)
	{
		_a = a._a;
		cout << "拷贝构造" << endl;
	}
	A& operator==(const A& a)
	{
		_a = a._a;
		cout << "赋值重载" << endl;
		return *this;
	}
private:
	int _a;
};

在这里插入图片描述

我们预想的是 :构造(100隐式类型转化)-> 拷贝构造(形参a),但是实际上编译器会直接优化成直接构造。


在这里插入图片描述

编译器只能在一行语句内进行优化,涉及多行语句编译器也不敢擅自进行优化。f2中的a先被定义了这个时候就会调用拷贝构造,返回的时候会产生临时变量这个时候就会发生拷贝构造。


在这里插入图片描述

正常来说应该是:直接构造(f2中的a1)-> 拷贝构造(产生的临时变量) -> 拷贝构造(返回给a1)。但是编译器会做优化处理,在返回的时候对a1直接进行拷贝构造,少调用一次拷贝构造。


在这里插入图片描述

这里使用匿名对象作为返回值的时候,a1接受的时候就被编译器优化成了对a1直接进行一次构造。这也是之前说匿名对象能够提升性能的原因。


简单总结:尽量使用引用传参,减少不必要的拷贝构造。尽量使用匿名对象作为返回值,同时接收返回值的时候采用拷贝构造的方式不要使用赋值接受。


7.总结

以上内容便是C++面向对象中的一些琐碎的零星知识点,以上内容如有问题,欢迎指正!

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

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

相关文章

如何在ONLYOFFICE v7.3中使用VSTACK和HSTACK公式

大家好&#xff0c;今天来给大家讲解一下&#xff0c;怎样在ONLYOFFICE v7.3中使用VSTACK和HSTACK公式&#xff0c; 这两个新公式也是ONLYOFFICE7.3版本更新后新的计算公式&#xff0c;请大家详细阅读本文。 ONLYOFFICE ONLYOFFICE文档是一款免费开源在线办公软件&#xff0c…

实验室装修|SICOLAB实验室装修方案

实验室装修不外乎通风与洁净的设计为重中之重&#xff0c;以下由SICOLAB介绍分享。一、实验室通风系统主要包含送风系统和排风系统两部分。&#xff08;1&#xff09;送风系统的设计需要考虑以下几个方面&#xff1a;1.1空气质量要求&#xff1a;实验室内的空气需要保持一定的洁…

生物素标记试剂1869922-24-6,Alkyne-PEG3-Biotin PC,炔烃PEG3生物素PC

1、试剂基团反应特点&#xff08;Reagent group reaction characteristics&#xff09;&#xff1a;PC alkyne-PEG3-Biotin含一个炔烃和一个 PEG 链接的可光裂解生物素基团。含 3 个单元 PEG 的 ADC linker&#xff0c;生物素本身是个游离的小分子&#xff0c;在生物实验中常常…

【汽车雷达通往自动驾驶的关键技术】

本文编辑&#xff1a;调皮哥的小助理 现代汽车雷达装置比手机还小&#xff0c;能探测前方、后方或侧方的盲点位置是否存在障碍物&#xff0c;但这还不百分之百实现全自动驾驶的。传统的汽车雷达分辨率都不高&#xff0c;只能“看到”一团东西&#xff0c;可以检测到汽车周围存在…

掌握使用yum为CentOS系统安装软件掌握使用apt为Ubuntu安装软件(扩展)

1.掌握使用yum为CentOS系统安装软件2.掌握使用apt为Ubuntu安装软件&#xff08;扩展&#xff09;1、Linux系统的应用商店 操作系统安装软件有许多种方式&#xff0c;一般分为&#xff1a; •下载安装包自行安装•如win系统使用exe文件、msi文件等•如mac系统使用dmg文件、pkg文…

容联七陌:ChatGPT大模型能力为智能客服带来新方向

科技云报道原创。 近几个月来&#xff0c;大众对ChatGPT预期的持续走高&#xff0c;也影响到了智能客服领域公司的命运。 一方面&#xff0c;ChatGPT的出现为智能客服场景带来了更加“智能”的可能性&#xff1b;但另一方面&#xff0c;有人认为ChatGPT完全可以替代现有的智能…

数值方法笔记3:线性和非线性方程组求解

前置知识1&#xff1a;矩阵范数前置知识2&#xff1a;舒尔补前置知识3&#xff1a;可约矩阵前置知识4&#xff1a;谱半径1.【线性方程组】直接求解&#xff1a;高斯消元法(LULULU分解)、LDVLDVLDV分解、LDLTLDL^TLDLT分解、UDUTUDU^TUDUT分解1.1 高斯消元法(LULULU分解)1.2 LDV…

公司缺人面了8个测试员,他们都有一个相同的缺点.....

年后公司缺人&#xff0c;面了不少测试&#xff0c;结果竟然没有一个合适的。 一开始瞄准的就是中级的水准&#xff0c;也没指望来大牛&#xff0c;提供的薪资在10-20k&#xff0c;面试的人很多&#xff0c;但平均水平很让人失望。 看简历很多都是3年工作经验&#xff0c;但面…

什么是深拷贝和浅拷贝?以及怎么实现深拷贝和浅拷贝?

拷贝浅是创建一个新对象&#xff0c;这个对象有着原始对象属性值的一份精确拷贝。如果属性是基本类型&#xff0c;拷贝的就是基本类型的值&#xff0c;如果属性是引用类型&#xff0c;拷贝的就是内存地址&#xff0c;所以如果其中一个对象改变了这个地址&#xff0c;就会影响到…

Excel操作-Apache-POI

Excel操作-Apache-POI一、场景需求将一些数据库信息导出为Excel表格将Excel表格数据导入数据库大量的数据的导入导出操作解决方案常用的解决方案&#xff1a;Apache POI与阿里巴巴easyExcelApache-POI介绍Apache-POI是基于office open XML标准&#xff08;OOXML&#xff09;和M…

合同审核机器人,提高合同审核效率,规避企业损失

官方网址&#xff1a;www.hanyunintel.com 在合同审核中面临的问题和挑战&#xff1a; 一、人工合同审核风险高&#xff1a; 人工合同审核依赖审核人员业务素养、精神状态、工作量、体力、工作态度及责任心&#xff0c;难免存在人工审核合同不严谨的情况&#xff0c;如果不能及…

【C++提高编程】C++全栈体系(二十一)

C提高编程 第三章 STL - 常用容器 三、deque容器 1. deque容器基本概念 功能&#xff1a; 双端数组&#xff0c;可以对头端进行插入删除操作 deque与vector区别&#xff1a; vector对于头部的插入删除效率低&#xff0c;数据量越大&#xff0c;效率越低deque相对而言&am…

简洁易用的记账小程序——微点记账

背景 由于每个月的信用卡账单太过吓人&#xff0c;记性也不是特别的好&#xff0c;加上微信支付宝账单中有些明细不是很明确。比如在京东花销的明细不会记录用户购买了什么&#xff0c;只会记录那个通道支出的。所以&#xff0c;才会有了想自己开发一款记账小程序&#xff0c;…

面试常问-Alpha测试和Beta测试

Alpha测试 Alpha测试是一种验收测试&#xff0c;在识别典型用户可能执行的任务并对其进行测试之前&#xff0c;执行该测试是为了识别所有可能的问题和错误。 尽可能简单地说&#xff0c;这种测试之所以被称为alpha&#xff0c;只是因为它是在软件开发的早期、接近开发结束时和…

使用loading动画让你的条件渲染页面更高级

前言在我们做项目的使用常常会使用条件渲染去有选择的给用户展示相关页面&#xff0c;如果渲染的数据或场景比较多比较复杂&#xff0c;那么往往需要3、4s的时间去完成&#xff0c;用户点击了之后就会陷入3、4s的空白期&#xff0c;并且这段时间屏幕是处于一种”未响应“的状态…

【欧拉筛法】洛谷 P3383 线性筛素数

3383. 线性筛素数 文章目录题目描述输入格式&#xff1a;输出格式&#xff1a;数据范围输入样例输出样例方法一&#xff1a;埃氏筛法解题思路代码复杂度分析&#xff1a;方法二&#xff1a;欧拉筛法解题思路代码复杂度分析&#xff1a;两种方法对比埃氏筛法欧拉筛法题目描述 给…

Jetpack管理生命周期——Lifecycle

Android Jetpack 对于任何一个产品来说&#xff0c;我们开发中都会面对哪些问题&#xff1f;如&#xff1a;产品交互、用户体验、代码结构、数据获取、数据存储、网络优化、任务调度等等&#xff0c;虽然在现在的阶段这些问题已经有了很好的解决和优化&#xff0c;也有很多大神…

堆球问题,开普勒猜想(格密码相关)

目录 一. 介绍 二. 历史进展分析 三.2维下的堆球问题 四. 3维下的堆球问题 五. 8维与24维下的堆球问题 总结 一. 介绍 堆球问题又叫堆球理论、最密堆积、球填充&#xff0c;英文为The Theory Of Sphere Packings。 堆球问题的本质就是填充一堆大小相同的球。要求这些球…

FANUC机器人通过KAREL程序实现与PLC位置坐标通信的具体方法示例

FANUC机器人通过KAREL程序实现与PLC位置坐标通信的具体方法示例 在通信IO点位数量足够的情况下,可以使用机器人的IO点传输位置数据,这里以传输机器人的实时位置为例进行说明。 基本流程如下图所示: 基本步骤可参考如下: 首先确认机器人控制柜已经安装了总线通信软件(例如…