C++11&QT复习 (十五)

发布于:2025-04-08 ⋅ 阅读:(31) ⋅ 点赞:(0)

Day10 标准模板库学习笔记(2025.04.02)

一、函数和数组能否放入 STL 容器?

1. 引用不能直接作为容器元素类型

C++ 的容器如 std::vector 不支持存储引用类型(例如 int&),因为引用并非对象本身,不能被复制或赋值。

std::vector<int&> rv; // ❌ 编译失败
✅ 推荐做法:使用 std::reference_wrapper<T>

标准库提供了 std::reference_wrapper 类模板,它可以将引用包装为对象,从而实现“间接”存储引用。

#include <functional>
std::vector<std::reference_wrapper<int>> rv1;
rv1.push_back(i); // i 是 int 类型变量

你也可以自定义类似的封装类,但通常没必要:

template <typename T>
class my_reference_wrapper {
  T& r;
public:
  my_reference_wrapper(T& i) : r(i) {}
};
2. 函数不能直接作为容器元素类型

函数名本身不是对象,而是函数类型,不能直接作为 std::vector 的元素类型。

std::vector<decltype(add)> fv; // ❌ 编译失败
✅ 推荐做法一:使用函数指针类型

函数名在赋值或传参时会自动“衰变”(decay)为函数指针。

auto f = add;  // 类型为 void(*)()
using func_ptr = decltype(f);

std::vector<func_ptr> fv1;
fv1.push_back(add);
fv1.push_back(minus);
✅ 推荐做法二:使用 std::function

std::function 是更通用、灵活的可调用对象封装器,可以容纳函数、函数指针、Lambda 表达式、函数对象等。

std::vector<std::function<void()>> fv2;
fv2.push_back(add);
fv2.push_back(minus);
fv2.push_back([](){});
fv2.push_back(Callable{}); // 假设 Callable 重载了 operator()
补充知识:函数和数组的“衰变”行为

C++ 中,数组和函数在很多场景下都会自动转换为指针类型,这种现象称为“衰变”(decay)。

int array[10];
auto p1 = array;     // int*
auto&& r1 = array;   // 引用仍保留 int[10] 类型

衰变行为可以用 <type_traits> 里的 std::decay_t<T> 模拟:

static_assert(std::is_same_v<decltype(array), std::remove_reference_t<decltype(r1)>>);
static_assert(std::is_same_v<decltype(p1), std::decay_t<decltype(array)>>);

函数也会衰变为函数指针:

static_assert(std::is_same_v<decltype(add), std::remove_reference_t<decltype(function_ref)>>);

二、数组能否放入容器?

❌ 直接存放原生数组类型 T[N] 不被支持
std::vector<int[10]> v;  // 编译失败

因为数组不能被拷贝或赋值。

✅ 推荐做法一:使用数组指针
std::vector<int*> v1;
v1.push_back(array);  // array 是 int[10]

但这种方式存在数组退化为指针的问题,可能引发不安全行为。

✅ 推荐做法二(更推荐):使用 std::array

std::array 是标准库提供的定长数组类型,兼具 C 数组的性能与 STL 容器的接口。

std::vector<std::array<int, 10>> v2;
v2.push_back(std::to_array(array)); // C++20 提供的安全转换

三、测试代码

#include <type_traits>
#include <functional>
#include <vector>
#include <algorithm>
#include <iostream>
#include <array>

void add()
{
  std::cout << "add" << std::endl;
}

void minus()
{
  std::cout << "minus" << std::endl;
}

// 普通类定义可以写在函数里,模板类不行
template <typename T>
class my_reference_wrapper
{
  T& r;
public:
  my_reference_wrapper(T& i) : r(i) {}
};

/*
容器无法容纳引用和函数?
*/
void container_can_not_contain_reference_and_function()
{
  int i = 1;
  int& ri = i;
  int& ri1 = i;
  // 想把这两个引用放到容器里

  // 想把上面两个函数 add 和 minus 放到容器里

  //std::vector<int&> rv; // 编译出错
  //std::vector<decltype(add)> fv; // 编译出错

  // 怎么办...

  // 对于引用,标准库提供了个包装类,把引用藏到自定义类型里
  std::vector<std::reference_wrapper<int>> rv1;
  rv1.push_back(ri);
  rv1.push_back(ri1);
  // 自己写也可以
  std::vector<my_reference_wrapper<int>> rv2;
  rv2.push_back(ri);
  rv2.push_back(ri1);
  // 你永远也不需要这么做...
  // 要么把对象拷贝进容器,要么移动进容器,把引用放进去是做什么???

  // 但是函数对象是确实有放到容器里的需求
  // 这里只是没写对元素的类型,试着拷贝函数给一个变量,看看变量的类型
  auto f = add; // f 是个函数指针
  using func_pointer = decltype(f);
  // 所以元素可以用函数指针类型
  std::vector<func_pointer> fv1;
  fv1.push_back(add);
  fv1.push_back(minus);

  // 标准库提供了更好的方案 std::function
  // 首先指定的还是函数类型,而不是函数指针类型
  // 其次接收可调用对象,即函数和重载了operator()的类都可以放进容器,包括lambda
  std::vector<std::function<decltype(add)>> fv2;
  fv2.push_back(add);
  fv2.push_back(minus);
  class Callable
  {
  public:
    void operator()() {}
  };
  fv2.push_back(Callable());
  fv2.push_back([]() {});

  // 发生了什么?为什么函数赋值后成了函数指针?
  // 这里涉及C++比较麻烦的地方,为了保持和C语言的兼容性
  // 所谓 decay 衰变
  // C语言有数组和函数,数组和函数在赋值/传参时会自动衰变为指针
  // 这是C编译器的行为,C++编译器为了兼容C,无奈照搬了,这也导致了C++的类型系统有瑕疵
  // 模板专家很多时候就是在处理这种奇怪的兼容性问题
  // 这也是为什么数组名是个指针,而函数自动会转为函数指针的原因
  int array[10];
  auto array_decay = array; // 类型衰变为 int*
  auto&& array_ref = array; // 万能引用可以看到精确类型,其实任何引用都可以
  auto function_decay = add; // 类型衰变为 void(*)()
  auto&& function_ref = add; // 万能引用可以看到精确类型,其实任何引用都可以
  static_assert(std::is_same_v<
    decltype(array_decay),
    std::decay_t<decltype(array)>>); // 自动decay和手动调用decay得到一样的类型
  static_assert(std::is_same_v<
    decltype(function_decay),
    std::decay_t<decltype(add)>>); // 自动decay和手动调用decay得到一样的类型
  static_assert(std::is_same_v<
    decltype(array),
    std::remove_reference_t<decltype(array_ref)>>); // 把引用移除就得到原来的类型
  static_assert(std::is_same_v<
    decltype(add),
    std::remove_reference_t<decltype(function_ref)>>); // 把引用移除就得到原来的类型

  // C数组会衰变,还允许做元素,C++这帮人...
  std::vector<int[10]> array_array;

  // 但,怎么塞元素...
  //array_array.push_back(array); // 编译出错

  // 完蛋,没法塞进去...
  // 都不让塞为什么允许C数组做元素啊...

  // 要么妥协,使用衰变类型,即指针
  std::vector<int*> array_array1;
  array_array1.push_back(array);

  // 要么改成标准库的定长数组std::array,推荐做法
  std::vector<std::array<int, 10>> array_array2;
  // 使用 std::to_array 从C数组构造标准库数组
  array_array2.push_back(std::to_array(array));

  // 记住
  // 永远不需要把引用放进容器
  // 函数放进容器是有需求的,推荐用std::function代替函数指针
  // 数组放进容器也有需求,推荐用std::array代替指针
  // 碰到问题先去找标准库的解决方案,不要自己(瞎)实现
}

四、总结

数据类型 是否能放入 STL 容器 推荐做法
引用 T& 使用 std::reference_wrapper<T>
函数 void() 使用 std::function<void()> 或函数指针
数组 T[N] 使用 std::array<T, N>T*

在这里插入图片描述

实践建议
  • 遇到“不能放入容器”的情况,优先查标准库是否已有合适的封装工具。
  • 避免自己写低质量、重复的封装类,标准库几乎总有更好的方案。
  • 多理解“类型衰变”在函数和数组上的行为,有助于掌握模板编程细节。

网站公告

今日签到

点亮在社区的每一天
去签到