【C++进阶知识】04 - 函数默认实参、默认初始化、initializer_list

1. 函数默认实参

默认实参需要注意以下几点:

(1)函数默认实参的赋值应从右往左,否则编译报错,因为参数入栈应该从右往左。

cpp 复制代码
void f(int, int, int = 1);
void f(int, int = 2, int);
void f(int = 3, int, int);

(2)类外的默认实参会使类的非默认构造函数变成默认构造函数。

cpp 复制代码
class A
{
public:

	A(int a);

	void Print()
	{
		std::cout << i << std::endl;
	}

	int i;
};
// 类外初始化默认实参
A::A(int a = 100) : i(a) {}
cpp 复制代码
/** 在类外初始化非默认构造函数,将其变为默认构造函数 */
A a = A();
a.Print();  // 输出:100

(3)如果在类中添加了该函数的该参数的默认实参,那么在类外再次定义该参数的默认实参,会发生重定义错误。

(4)虚函数的默认实参将根据对象的静态类型(编译时直接指定不会更改的类型)确定。

cpp 复制代码
struct F
{
	virtual ~F()
	{
		// 父类
	}
};

struct C : F
{
	// 子类
};

/**
* 对于p来说静态类型就是F
* 对于p来说动态类型就是C
* 所以如果父类和子类都有默认实参的话,会使用F中的默认实参函数
*/
F* p = new C();

2. 默认初始化

默认初始化没什么难的,需要注意的是默认初始化是C++11新添加的,主要看一下位域初始化。

cpp 复制代码
struct B
{
	// int的低8位被初始化为12
	int x : 8 = 12;
	// int的低8位被初始化为17
	int y : 4 { 17 };
};

在使用位域初始化的时候,一定要注意后面使用的运算符与:的优先级问题。

3 initializer_list

3.1 初始化列表的本质

cpp 复制代码
#include <initializer_list>
std::initializer_list

template <class _Elem>
class initializer_list {
public:
    using value_type      = _Elem;
    using reference       = const _Elem&;
    using const_reference = const _Elem&;
    using size_type       = size_t;

    using iterator       = const _Elem*;
    using const_iterator = const _Elem*;

    constexpr initializer_list() noexcept : _First(nullptr), _Last(nullptr) {}

    constexpr initializer_list(const _Elem* _First_arg, const _Elem* _Last_arg) noexcept
        : _First(_First_arg), _Last(_Last_arg) {}

    _NODISCARD constexpr const _Elem* begin() const noexcept {
        return _First;
    }

    _NODISCARD constexpr const _Elem* end() const noexcept {
        return _Last;
    }

    _NODISCARD constexpr size_t size() const noexcept {
        return static_cast<size_t>(_Last - _First);
    }

private:
    const _Elem* _First;
    const _Elem* _Last;
};

可以看出initializer_list就是一个有begin和end的一片内存空间。

cpp 复制代码
int x[] = { 1, 2, 3, 4, 5 };
std::vector<int> v{1, 2, 3, 4, 5};

相当于使用initializer_list{1, 2, 3, 4, 5},就是先构造了一个array{ 1, 2, 3, 4, 5 },再把首地址和尾地址赋给begin和end。

cpp 复制代码
class A
{
public:
	/** 使用初始化列表构造并遍历 */
	A(std::initializer_list<int> list)
	{
		for (const int* item = list.begin(); item != list.end(); ++item)
		{
			std::cout << *item << std::endl;
		}
	}
};

初始化优先级:

cpp 复制代码
/** 调用构造5个元素,每个元素都是5 */
std::vector<int> x1(5, 5);
/** 调用构造2个元素,5和5 */
std::vector<int> x2{5, 5};

3.2 隐式缩窄转换

隐式缩窄转换规则:
(1)高位向低位转换,如double向float,float向int。
(2)从整数类型向超过其最大值的类型转换,如:int a = 999,向char转换。

3.3 指定初始化

为了增加灵活性,C++20增加了指定初始化。

cpp 复制代码
struct Point3D
{
	int x;
	int y;
	int z;
};
// 初始化列表构造,x=0,y=0,z=3
Point3D{.z = 3};

虽然增加了指定初始化,但有很多的限定:
(1)Point3D如果有了构造函数,则初始化列表会按照构造函数进行,指定的成员变量很有可能失败。
(2)指定初始化的顺序要按照定义顺序进行。
(3)联合体一次只能指定一个;指定初始化不能嵌套;指定初始化不能和普通的混用。

相关推荐
坚果派·白晓明41 分钟前
【鸿蒙PC】SDL3 适配:AtomCode + Skills 快速集成 NAPI 测试工具
c++·华为·ai编程·harmonyos·atomcode
huangdong_1 小时前
淘宝商品SKU图自动分类技术深度解析:从DOM解析到智能归档
开发语言·javascript·ecmascript
阿正的梦工坊1 小时前
【Rust】12-借用检查器与非词法生命周期
开发语言·后端·rust
qq_2518364571 小时前
基于java Web网络订餐系统设计与实现 源码文档
java·开发语言·前端
秋91 小时前
3年经验Python后端转AI Engineer:3个月实战转型计划(2026版)
开发语言·人工智能·python
凡人叶枫1 小时前
Effective C++ 条款17:以独立语句将 newed 对象置入智能指针
java·linux·开发语言·c++·算法
飞天狗1112 小时前
零基础JavaWeb入门——第2课:让网页“活”起来 —— JSP是什么?
java·开发语言·前端·后端·web
醇氧2 小时前
【Linux】Java 服务生产级部署指南:实现常驻后台、开机自启与系统服务化管理
java·开发语言
凡人叶枫3 小时前
Effective C++ 条款16:成对使用 new 和 delete 时要采取相同形式
开发语言·c++·effective c++
不吃土豆的马铃薯3 小时前
C++ 高性能网络缓冲区 Buffer 源码解析
linux·服务器·开发语言·网络·c++