STL-适配器(面试复习4)

目录

[C++ STL 适配器(Adapters)高频面试题整理版](#C++ STL 适配器(Adapters)高频面试题整理版)

一、基础概念类(必考)

[1️⃣ 什么是 STL 适配器?分为哪几类?](#1️⃣ 什么是 STL 适配器?分为哪几类?)

[二、容器适配器(🔥 核心重点)](#二、容器适配器(🔥 核心重点))

[2️⃣ stack / queue 的默认底层容器是什么?为什么?](#2️⃣ stack / queue 的默认底层容器是什么?为什么?)

[3️⃣ priority_queue 的底层是什么?](#3️⃣ priority_queue 的底层是什么?)

[4️⃣ 为什么 stack / queue 不提供迭代器?](#4️⃣ 为什么 stack / queue 不提供迭代器?)

[5️⃣ 如何指定 stack / queue 的底层容器?](#5️⃣ 如何指定 stack / queue 的底层容器?)

stack

queue

三、算法与复杂度(必背)

[6️⃣ priority_queue 的时间复杂度](#6️⃣ priority_queue 的时间复杂度)

[7️⃣ 如何实现小顶堆?](#7️⃣ 如何实现小顶堆?)

四、进阶对比(加分项)

[8️⃣ push() vs emplace()](#8️⃣ push() vs emplace())

[9️⃣ deque vs vector 作为 stack 底层对比](#9️⃣ deque vs vector 作为 stack 底层对比)

五、迭代器适配器(理解型)

[🔟 reverse_iterator 的底层原理](#🔟 reverse_iterator 的底层原理)

[为什么 rbegin() == end()?](#为什么 rbegin() == end()?)

[✅ 面试一句话总结](#✅ 面试一句话总结)


C++ STL 适配器(Adapters)高频面试题整理版

核心一句话

STL 适配器不是新容器/算法,而是对已有组件的接口封装与限制 ,体现的是适配器设计模式(Adapter Pattern)


一、基础概念类(必考)

1️⃣ 什么是 STL 适配器?分为哪几类?

定义:

STL 适配器是一种设计模式,通过封装已有的容器 / 迭代器 / 函数对象,对其接口进行转换,使其符合特定使用场景。

三大类:

  1. 容器适配器(Container Adapters)

    • std::stack

    • std::queue

    • std::priority_queue

  2. 迭代器适配器(Iterator Adapters)

    • std::reverse_iterator

    • std::back_insert_iterator

    • std::front_insert_iterator

    • std::insert_iterator

  3. 函数适配器(Function Adapters)

    • std::bind

    • std::function

    • std::not1 / std::not2(C++11 后已废弃,推荐 lambda)


二、容器适配器(🔥 核心重点)

2️⃣ stack / queue 的默认底层容器是什么?为什么?

默认底层容器:

复制代码
std::deque

原因分析:

对比点 deque vector list
头尾插删 O(1) 头部 O(n) O(1)
扩容成本 分段扩容,无整体拷贝 扩容需整体拷贝
内存局部性 较好 最好 最差
额外指针

👉 结论
deque性能稳定性 + 接口适配性上最均衡,因此成为默认选择。


3️⃣ priority_queue 的底层是什么?

  • 数据结构:堆(Heap)

    • 默认:大顶堆(Max Heap)
  • 底层容器std::vector

原因:

  • 堆是完全二叉树

  • 使用数组 / vector 可通过下标快速定位:

    • 左孩子:2*i + 1

    • 右孩子:2*i + 2


4️⃣ 为什么 stack / queue 不提供迭代器?

核心原因:维护抽象语义

  • stack:LIFO(后进先出)

  • queue:FIFO(先进先出)

如果提供迭代器:

  • 用户可访问中间元素

  • 可破坏数据结构语义

  • 与"只能在端点操作"的设计目标冲突

👉 这是"接口约束",不是能力不足


5️⃣ 如何指定 stack / queue 的底层容器?

容器适配器是模板类,可指定第二个模板参数。

stack

要求容器支持:

  • push_back

  • pop_back

  • back

    std::stack<int, std::vector<int>> s;

可选容器:

  • vector

  • deque(默认)

  • list


queue

要求容器支持:

  • push_back

  • pop_front

  • front

  • back

    std::queue<int, std::list<int>> q;

⚠️ vector ❌(没有 pop_front


三、算法与复杂度(必背)

6️⃣ priority_queue 的时间复杂度

操作 复杂度 原因
push() O(log n) 向上调整(heapify up)
pop() O(log n) 向下调整(heapify down)
top() O(1) 直接访问堆顶

7️⃣ 如何实现小顶堆?

修改第三个模板参数(比较器):

复制代码
std::priority_queue<
    int,
    std::vector<int>,
    std::greater<int>
> minHeap;
  • 默认:std::less<T> → 大顶堆

  • 使用:std::greater<T> → 小顶堆


四、进阶对比(加分项)

8️⃣ push() vs emplace()

对比 push emplace
构造位置 容器外 容器内
临时对象 可能有 没有
拷贝 / 移动
性能 稍差 更优

👉 推荐:优先使用 emplace()


9️⃣ deque vs vector 作为 stack 底层对比

维度 deque(默认) vector
扩容 分段,无整体拷贝 翻倍扩容
内存连续 伪连续 严格连续
空间浪费 少量 buffer 最多 ~50%
适用场景 通用、大对象 小对象、极致局部性

五、迭代器适配器(理解型)

🔟 reverse_iterator 的底层原理

本质:对正向迭代器的封装

  • ++rit → 实际执行 --it

  • --rit → 实际执行 ++it

为什么 rbegin() == end()

STL 区间是左闭右开 [begin, end)

复制代码
rbegin() = reverse_iterator(end())

解引用逻辑:

复制代码
*rit  等价于  *(it - 1)

👉 这是反向迭代器最容易被问的陷阱点


✅ 面试一句话总结

STL 适配器通过限制接口而不是增强功能

保证数据结构的语义正确性,

是 STL 设计哲学中**"抽象与约束"**的典型体现。

相关推荐
Han.miracle2 小时前
《Spring MVC 响应机制综合实践:页面、数据、JSON 与响应配置》
java·spring·springboot
xiaoxue..2 小时前
列表转树结构:从扁平列表到层级森林
前端·javascript·算法·面试
JHC0000002 小时前
dy直播间评论保存插件
java·后端·python·spring cloud·信息可视化
SuperherRo2 小时前
JAVA攻防-FastJson专题&面试不出网利用&BCEL字节码&C3P0二次&Impl链&延时判断
java·fastjson·不出网
TH_12 小时前
18、删除WPSOfficeWord文档中的空白页
java
一雨方知深秋2 小时前
数组定义及访问
java·数组·二维数组·for·length·定义访问
alanesnape2 小时前
Java异常处理详解:Exception、ArithmeticException、FileNotFoundException
java·开发语言
while(1){yan}2 小时前
数据链路层与物理层
java·网络·网络协议
野蛮人6号2 小时前
黑马微服务 p23Docker02 docker的安装 如何正确安装docker,黑马微服务给的文档不行了,如何正确找到解决方法
java·docker·微服务·架构