当前位置: 首页 > news >正文

类和对象(四)

构造函数中的初始化列表

之前在实现构造函数时,主要是在函数体内进行赋值,而构造函数还有另一种初始化方式,通过初始化列表进行初始化。

初始化列表的使⽤⽅式是以⼀个冒号开始,接着是⼀个以逗号分隔的数据成员列表,每个"成员变量"后⾯跟⼀个放在括号中的初始值或表达式。每个成员变量在初始化列表中只能出现⼀次,语法理解上初始化列表可以认为是每个成员变量定义初始化的地⽅。

要注意的是,初始化列表中按照成员变量在类中声明顺序进行初始化,跟成员在初始化列表出现的的先后顺序是无关的。但一般建议声明顺序和初始化列表顺序保持⼀致。

#include<iostream>class Date
{
public:Date(int year = 1, int month = 1, int day = 1):_year(year), _month(month), _day(day){}void Print(){std::cout << _year << "-" << _month << "-" << _day << std::endl;}private:int _year;int _month;int _day;
};int main()
{Date d1(2024, 7, 20);d1.Print();return 0;
}

要区分,函数参数中给的缺省值并不能初始化成员变量。两者是不同的东西,要加一区分。

#include<iostream>class Date
{
public:Date(int year = 1, int month = 1, int day = 1):_year(year), _month(month){}void Print(){std::cout << _year << "/" << _month << "/" << _day << std::endl;}private:int _year;int _month;int _day;
};int main()
{Date d1(2024, 7, 20);d1.Print();return 0;
}

这里 _day 并未初始化,为随机值。

#include<iostream>class Date
{
public:Date(int year = 1, int month = 1, int day = 1):_year(year), _month(month){// _day未在初始化列表中初始化,在函数体内也能初始化,如果都没有就为随机值_day = day;}void Print(){std::cout << _year << "/" << _month << "/" << _day << std::endl;}private:int _year;int _month;int _day;
};int main()
{Date d1(2024, 7, 20);d1.Print();return 0;
}

构造函数在初始化时,会先走初始化列表进行初始化,若初始化列表没有初始化,再走构造函数内部看是否初始化,若都没有,就只能要看编译器是否初始化,但一般都为随机值。

一般的内置类型的成员普通的构造函数就可以实现,不需要初始化列表。但有些成员变量只能通过初始化列表进行初始化。

前面说过,初始化列表可以认为是每个成员变量定义初始化的地方,而有些变量只能在定义的时候进行初始化,如:

// const常量只能在定义时初始化一次
const int a = 1;
//引用也只能在初始化时定义
int x = 10;
int& y = x;

还有一个,没有默认构造的类类型变量。如:

该默认构造无需传参,其初始化时若无参数,则会使用缺省值,故可完成初始化。但如果没有缺省值,则程序会编译出错。

#include<iostream>class A
{
public:A(int a):_a(a){std::cout << "A()" << std::endl;}private:int _a;
};class Date
{
public:Date(int& xx, int year = 1, int month = 1, int day = 1):_year(year), _month(month),_day(day),a(1),x(xx){}void Print(){std::cout << _year << "/" << _month << "/" << _day << std::endl;}private:int _year;int _month;int _day;const int a;int& x;A y;
};int main()
{int x = 1;Date d1(x, 2024, 7, 20);return 0;
}

其编译会报错

这里必须要传入参数,所以我们就需要在初始化列表进行初始化。

#include<iostream>class A
{
public:A(int a):_a(a){std::cout << "A()" << std::endl;}private:int _a;
};class Date
{
public:Date(int& xx, int year = 1, int month = 1, int day = 1):_year(year), _month(month),_day(day),a(1),x(xx),y(1){}void Print(){std::cout << _year << "/" << _month << "/" << _day << std::endl;}private:int _year;int _month;int _day;const int a;int& x;A y;
};int main()
{int x = 1;Date d1(x, 2024, 7, 20);return 0;
}

此外,C++11⽀持在成员变量声明的位置给缺省值,这个缺省值主要是给没有显示在初始化列表初始化的成员使⽤的。

#include<iostream>class A
{
public:A(int a):_a(a){std::cout << "A()" << std::endl;}private:int _a;
};class Date
{
public:Date(int& xx, int year = 1, int month = 1, int day = 1):_year(year), _month(month),_day(day),x(xx){}void Print(){std::cout << _year << "/" << _month << "/" << _day << std::endl;}private:int _year;int _month;int _day;const int a = 1;int& x;A y = 1;
};int main()
{int x = 1;Date d1(x, 2024, 7, 20);return 0;
}

但尽量使⽤初始化列表初始化,因为那些你不在初始化列表初始化的成员也会⾛初始化列表,如果这 个成员在声明位置给了缺省值,初始化列表会用这个缺省值初始化。

但如果没有给缺省值,对于没有显示在初始化列表初始化的内置类型成员是否初始化取决于编译器,C++并没有规定。对于没有显示在初始化列表初始化的自定义类型成员会调用这个成员类型的默认构造函数,如果没有默认构造会编译错误。所以尽量使用初始化列表。


类型转换

举最简单的例子来说:

#include<iostream>int main()
{int a = 10;double b = a;return 0;
}

这是一个简单的类型转换。

C++⽀持内置类型隐式类型转换为类类型对象,但需要有相关内置类型为参数的构造函数。

#include<iostream>class A
{
public://构造函数A(int a){_a = a;}void Print(){std::cout << _a  << std::endl;}
private:int _a;
};int main()
{// 6构造⼀个A的临时对象,再用这个临时对象拷贝构造a1// 但若编译器遇到连续构造+拷贝构造就会优化为直接构造A a1 = 6;a1.Print();//这个隐式类型转换过程为://构造函数A a2(1);//拷贝构造,注:该类只有内置类型的成员变量,编译器自动生成的拷贝构造就能实现目的A a3 = a2;//所以就直接通过隐式类型转换来使用return 0;
}

在C++11之前,其是不支持多参数转化,一些较老的编译器就不支持多参数转化。

#include<iostream>class A
{
public://构造函数A(int a1 = 1, int a2 = 2){_a1 = a1;_a2 = a2;}void Print(){std::cout << _a1 << " " << _a2 << std::endl;}private:int _a1;int _a2;
};int main()
{A a = { 6,66 };a.Print();//相当于:A a1(1, 11);A a2 = a1;a2.Print();return 0;
}

构造函数前⾯加关键字 explicit 就可以不再⽀持隐式类型转换。


static成员

⽤static修饰的成员变量,称之为静态成员变量,静态成员变量⼀定要在类外进⾏初始化。其为所有类对象所共享,不属于某个具体的对象,不存在对象中,存放在静态区。

#include<iostream>class A
{
public:private:int a;static int b;
};//在类外初始化静态成员变量
int A::b = 1;int main()
{return 0;
}

而⽤static修饰的成员函数,称之为静态成员函数,静态成员函数是没有this指针的。静态成员函数可以访问其他的静态成员,但是不能访问⾮静态的,因为没有this指针。但⾮静态的成员函数,可以访问任意的静态成员变量和静态成员函数。

#include<iostream>class A
{
public:static int Get(){return b;}void func(){std::cout << a << ' ' << b << std::endl;}private:int a = 1;static int b;
};//在类外初始化静态成员变量
int A::b = 1;int main()
{A a;a.func();return 0;
}

一定要注意,静态成员变量不能在声明位置给缺省值初始化,因为缺省值是个构造函数初始化列表的,静态成员变量不属于某个对象,不⾛构造函数初始化列表。

由于静态成员也是类的成员,所以其受public、protected、private等访问限定符和类域的限制。

相关文章:

  • 北京网站建设多少钱?
  • 辽宁网页制作哪家好_网站建设
  • 高端品牌网站建设_汉中网站制作
  • AV1技术学习:Quantization
  • centos系统mysql主从复制(一主一从)
  • 全国区块链职业技能大赛第九套区块链产品需求分析与方案设计
  • 腾讯云开发者《中国数据库前世今生》有奖创作季
  • Linux grep技巧 提取log中的json数据
  • MySQL第四次作业
  • Try ubuntu core (by quqi99)
  • leetcode日记(48)排列序列
  • Harmony Next -- 通用标题栏:高度自定义,可设置沉浸式状态,正常状态下为:左侧返回、居中标题,左中右均可自定义视图。
  • 「运费速查神器」精明买家必备!一键查询1688供应商发货费用
  • 白骑士的PyCharm教学基础篇 1.2 PyCharm基本操作
  • VMware三种网络模式---巨细
  • Linux下如何使用GitLab进行团队协作
  • ES6 数值的扩展(十八)
  • 【MySQL】:对库和表的基本操作方法
  • 〔开发系列〕一次关于小程序开发的深度总结
  • cookie和session
  • docker python 配置
  • JDK9: 集成 Jshell 和 Maven 项目.
  • Lucene解析 - 基本概念
  • MaxCompute访问TableStore(OTS) 数据
  • React组件设计模式(一)
  • vue:响应原理
  • 番外篇1:在Windows环境下安装JDK
  • 干货 | 以太坊Mist负责人教你建立无服务器应用
  • 关于Java中分层中遇到的一些问题
  • 关于字符编码你应该知道的事情
  • 来,膜拜下android roadmap,强大的执行力
  • 力扣(LeetCode)21
  • 聊聊spring cloud的LoadBalancerAutoConfiguration
  • 如何设计一个比特币钱包服务
  • 什么软件可以提取视频中的音频制作成手机铃声
  • 微服务框架lagom
  • 用mpvue开发微信小程序
  • d²y/dx²; 偏导数问题 请问f1 f2是什么意思
  • 数据可视化之下发图实践
  • ​MPV,汽车产品里一个特殊品类的进化过程
  • # 详解 JS 中的事件循环、宏/微任务、Primise对象、定时器函数,以及其在工作中的应用和注意事项
  • #Linux(帮助手册)
  • $().each和$.each的区别
  • (1)(1.19) TeraRanger One/EVO测距仪
  • (14)目标检测_SSD训练代码基于pytorch搭建代码
  • (delphi11最新学习资料) Object Pascal 学习笔记---第13章第6节 (嵌套的Finally代码块)
  • (备忘)Java Map 遍历
  • (官网安装) 基于CentOS 7安装MangoDB和MangoDB Shell
  • (全注解开发)学习Spring-MVC的第三天
  • (三) prometheus + grafana + alertmanager 配置Redis监控
  • (四)Android布局类型(线性布局LinearLayout)
  • (心得)获取一个数二进制序列中所有的偶数位和奇数位, 分别输出二进制序列。
  • (循环依赖问题)学习spring的第九天
  • (一)插入排序
  • (转)jdk与jre的区别
  • (转)拼包函数及网络封包的异常处理(含代码)
  • (转载)Google Chrome调试JS
  • ./indexer: error while loading shared libraries: libmysqlclient.so.18: cannot open shared object fil