在当前竞争激烈的技术领域,掌握 C++ 不仅是一项核心技能,更是迈向高薪职位的敲门砖。作为被广泛应用于系统开发、游戏引擎、嵌入式系统和高性能计算、自动驾驶的语言,C++ 的强大性能和灵活性让它在面试中占据了重要地位。如果你正在为迎接 2025 年的求职挑战做准备,那么刷 C++ 面试题绝对是 必不可少的一环!本文从基础面试、面向对象编程、STL容器精选了25道面试题。

为了让你的学习之旅更加高效,知识分享官还精心准备了一系列优质资源:

  • 源码小游戏:从实践中掌握核心技术

  • 经典书籍推荐:构建扎实的理论基础

  • 技术大会 PPT:深度剖析高级应用场景

从基础概念到高级实践,全面覆盖 C++ 技术关键点,助你轻松应对面试,脱颖而出,迈向心仪的高薪职位!


接下来是25道经典面试题的解析,大家也可以直接在资料包中查看。

基础部分

1、引用和指针之间的区别?

引用和指针是 C++ 中两种重要的间接访问机制。引用实质上是一个别名,它在声明时必须初始化,一旦绑定就不能改变。引用不能为空,也不能建立引用的引用,这使得它比指针更安全。引用在内存中实际上是作为一个常量指针实现的,但它对程序员隐藏了指针的复杂性。指针则是一个变量,它存储了另一个变量的内存地址。指针可以为空,可以改变它所指向的对象,也可以进行指针运算。指针的灵活性使它成为一个强大的工具,但同时也容易导致程序错误,如悬空指针和内存泄漏。

2、堆栈和堆中的内存分配有何区别?

栈内存的分配和释放是由编译器自动完成的。当变量超出其作用域时,栈内存会自动释放。栈内存的分配和释放速度很快,因为它使用简单的指针移动来实现。栈内存的大小是固定的,在程序运行时就已确定。

堆内存则是动态分配的,由程序员手动管理或使用智能指针等工具管理。堆内存的生命周期不受作用域限制,可以在需要时分配,在不需要时释放。堆内存的大小理论上只受系统可用内存的限制。但堆内存的分配和释放相对较慢,且容易产生内存碎片。

3、存在哪些类型的智能指针?

智能指针类型:C++11 定义了三种智能指针。

  • unique_ptr 实现了独占式拥有概念,它保证一个对象只被一个指针拥有

  • shared_ptr 允许多个指针指向同一个对象,通过引用计数机制来管理对象的生命周期

  • weak_ptr 是一种弱引用,它不会增加引用计数,主要用于解决 shared_ptr可能产生的循环引用问题

  • auto_ptr 是 C++98 引入的智能指针,但由于其危险的拷贝语义已在 C++11中被弃用

每种智能指针都有其特定的使用场景,选择合适的智能指针可以大大降低内存管理的复杂性。

4、unique_ptr 是如何实现的?我们如何强制在 unique_ptr 中仅存在一个对象所有者?

unique_ptr 的核心思想是独占式拥有。它通过删除拷贝构造函数和拷贝赋值运算符来保证一个对象只能被一个 unique_ptr 拥有。它支持移动语义,允许在保证安全的情况下转移对象的所有权。

unique_ptr 内部包含一个原始指针和一个删除器。删除器是一个可调用对象,负责在 unique_ptr 析构时释放所管理的资源。unique_ptr 的实现非常轻量,在大多数情况下不会带来任何额外的开销。

5、shared_ptr 如何工作?对象之间如何同步引用计数器?

shared_ptr 使用引用计数来追踪有多少个 shared_ptr 共享同一个对象。当引用计数增加时(比如拷贝构造),计数器加一;当引用计数减少时(比如 shared_ptr 析构),计数器减一。当计数器降为零时,管理的对象会被删除。

为了保证线程安全,引用计数的更新必须是原子操作。shared_ptr 内部实际上包含两个指针:一个指向管理的对象,另一个指向控制块。控制块包含引用计数、删除器等信息。这种实现方式使 shared_ptr 的大小是原始指针的两倍。

6、我们可以复制 unique_ptr 或者将其从一个对象传递到另一个对象吗?

由于 unique_ptr 强制独占所有权,所以它不能被复制。但它可以通过 std::move 来转移所有权。在转移后,原来的 unique_ptr 会变为空指针,而目标 unique_ptr 获得对象的所有权。

这种设计使得 unique_ptr 可以方便地在函数间传递对象的所有权。例如,factory 函数可以返回一个 unique_ptr,调用者就获得了返回对象的所有权。unique_ptr 也可以存储在容器中,但必须使用移动语义来操作。

7、什么是右值和左值?

在 C++ 中,左值是一个位置值,它标识一个持久的对象。左值有一个地址,可以取地址操作符作用于它。典型的左值包括变量名、解引用的指针等。

右值表示临时值,它不能取地址。右值可以分为纯右值(pr-value)和将亡值(x-value)。纯右值是临时的、不具名的值,如字面常量。将亡值是即将被销毁的值,比如即将被移动的对象。

8、进程和线程之间的区别?

进程(Process)是一个程序的运行实例。它包含程序代码和运行时所需的资源(如内存、文件句柄等)。进程之间是相互独立的,通常由操作系统进行管理。

线程(Thread)是进程中的一个执行路径。一个进程可以包含多个线程,它们共享该进程的资源(如内存地址空间、文件句柄等)。线程之间的关系更紧密, 通信效率更高。

每个进程有自己的独立资源,包括内存空间、堆栈、全局变量等。进程间的通信(如通过管道、消息队列或共享内存)通常比线程间通信更复杂且开销更大。 线程共享进程的资源,如内存空间和文件句柄。因此,线程之间的通信更简单,通常可以通过共享内存直接进行,但需要额外考虑线程安全问题。

9、同一个线程可以重复运行吗?

在 C++ 中,同一个线程对象不能重复运行。一旦线程运行完成(即线程的 join() 或 detach() 操作结束),对应的 std::thread 对象会进入一个“不可再运行”的状态。如果你尝试重复启动同一个线程对象,会导致程序崩溃或抛出异常。

10、同步线程的方法?

在 C++ 中,线程同步是指协调多个线程对共享资源的访问,避免数据竞争和不一致的行为。C++ 提供了多种方式来实现线程同步,包括锁、条件变量、原子操作等。以下是常见的同步方法及其使用示例,可以基于不同的使用场景来选择,分别是使用互斥锁 (std::mutex)、使用读写锁 (std::shared_mutex)、

条件变量 (std::condition_variable)、原子操作 (std::atomic)、屏障同步 (std::barrier) (C++20 引入)。

11、什么是死锁?

死锁是一种状态,指两个或多个线程因为相互等待对方释放资源,导致所有线程都无法继续执行的现象。

简单来说:线程 A 等待线程 B,线程 B 又在等待线程 A,结果谁都动不了。

根据操作系统理论,死锁的发生需要满足以下四个条件:

  • 互斥:资源只能被一个线程独占。

  • 持有并等待:一个线程已经持有某些资源,同时等待其他线程释放资源。

  • 不可剥夺:资源不能被强制剥夺,只能由持有线程主动释放。

  • 循环等待:存在线程之间形成的资源循环等待链。

12、什么是 std::move 和 std::forward()

std::move 和 std::forward 是 C++ 标准库中的两个函数模板,主要用于优化对象的传递和资源管理,特别是在涉及右值引用和完美转发时。

std::move 用于显式地将对象转化为右值引用,从而允许移动语义的使用。它不会移动对象的内容,只是把对象的“身份”从左值变成右值。

std::forward 用于完美转发,将传递给函数的参数保持其原始的左值或右值性质。常与模板函数和万能引用(T&&)搭配使用。

面向对象编程(OOP)

1、访问某些类的私有字段的方法?

在 C++ 中,有几种合法的方式可以访问类的私有成员。最常用的是通过友元(friend)声明,可以是友元函数或友元类。友元声明允许外部代码访问类的私有成员。

另一种方式是通过类的公共接口方法来间接访问私有成员。这是面向对象设计中推荐的方式,因为它维护了类的封装性。还可以使用嵌套类,因为嵌套类可以访问外部类的所有成员。

2、一个类可以继承多个类吗?

C++ 支持多重继承,即一个类可以同时继承多个基类。但这可能导致菱形继承问题,即一个类通过不同的路径继承了同一个基类的多个实例。

为了解决这个问题,C++ 引入了虚继承。虚继承确保共同基类只有一个实例。虽然 C++ 支持多重继承,但在实际开发中应该谨慎使用,因为它可能增加代码的复杂性。

3、静态字段是否在类构造函数中初始化?

静态成员变量属于类而不是对象,它们不在构造函数中初始化。静态成员需要在类外进行定义和初始化,除非是整型或枚举类型的 const static 成员,这种情况可以在类内初始化。

这样设计的原因是为了避免静态成员被多次初始化。在程序的整个生命周期中,静态成员只需要初始化一次。静态成员的初始化发生在程序开始执行之前。

4、构造函数/析构函数中会抛出异常吗?如何防止这种情况发生?

构造函数可以抛出异常,这通常用于指示初始化失败。但析构函数应该避免抛出异常,因为如果在异常处理过程中析构函数抛出异常,程序会立即终止。

为了防止构造函数抛出异常,可以使用 try-catch 块包装可能抛出异常的代码,或者使用初始化列表来确保资源获取。对于析构函数,应该将其声明为noexcept,并在内部处理所有可能的异常。

5、什么是虚方法?

虚方法是 C++ 实现运行时多态的机制。当基类中声明一个函数为 virtual 时,派生类可以重写这个函数。在运行时,程序会根据对象的实际类型来调用适当的函数版本。

虚函数通过虚函数表(vtable)来实现。每个包含虚函数的类都有一个 vtable,其中存储了该类虚函数的地址。每个对象都包含一个指向 vtable 的指针(vptr)。这种机制允许在运行时动态绑定函数调用。

6、为什么我们需要虚拟析构函数?

当通过基类指针删除派生类对象时,如果析构函数不是虚函数,则只会调用基类的析构函数,而不会调用派生类的析构函数。这会导致资源泄漏。

因此,当一个类可能作为基类时,其析构函数应该声明为虚函数。这确保了在删除对象时,无论使用什么类型的指针,都能正确调用整个继承链上的析构函数。

7、抽象类和接口的区别?

抽象类是包含至少一个纯虚函数的类。它不能被实例化,只能作为基类使用。抽象类可以包含普通成员函数和数据成员,这些可以在派生类中直接使用。

接口是一种特殊的抽象类,它只包含纯虚函数。在 C++ 中,接口通常被实现为所有函数都是纯虚函数的抽象类。接口定义了一个对象能够做什么,而不规定如何做。

8、构造函数可以是虚拟的吗?

构造函数不能是虚函数。这是因为在调用构造函数时,对象还没有被完全构造,vptr 还没有被初始化。因此不能使用虚函数机制。

如果需要根据运行时条件创建不同类型的对象,可以使用工厂模式。工厂模式通过一个静态成员函数来创建对象,这个函数可以根据参数返回不同类型的对象。

9、关键字 const 如何用于类方法?

const 成员函数承诺不会修改对象的状态。它们可以被 const 对象调用,这提供了一种编译时的类型安全机制。const 成员函数不能调用非 const 成员函数。如果需要在 const 成员函数中修改某些成员变量,可以将这些变量声明为mutable。mutable 允许在 const 成员函数中修改特定的成员变量,这通常用于缓存等不影响对象逻辑状态的场合。

10、如何保护对象不被复制?

有几种方式可以防止对象被复制。最现代的方式是使用 =delete 来删除拷贝构造函数和拷贝赋值运算符。这会在编译时阻止任何复制操作。

另一种方式是将拷贝构造函数和拷贝赋值运算符声明为 private,并且不提供实现。这也可以防止复制,但错误信息可能不如 =delete 清晰。

STL 容器

1、vector 和 list 之间的区别?

vector 是一种连续存储的容器,它在内存中分配一块连续的空间。这使得 vector 支持随机访问,但在中间插入或删除元素时需要移动后续元素。当 vector 需要更多空间时,它会重新分配一个更大的连续空间。

list 是一种链式存储的容器,它的元素可以分散在内存的不同位置。list 不支持随机访问,但在任何位置插入或删除元素都很快,因为只需要修改相关节点的指针。list 的每个元素都需要额外的内存来存储指针。

2、map 和 unordered_map 之间的区别?

map 基于红黑树实现,它保持键值对按键的顺序存储。这使得 map 的查找、插入和删除操作的时间复杂度都是O(log n)。map 占用的内存较少,而且可以按序遍历。

unordered_map 基于哈希表实现,它不保持任何顺序。在理想情况下,查找、插入和删除操作的时间复杂度都是 O(1)。但 unordered_map 需要额外的内存来存储哈希表,而且可能需要处理哈希冲突。

3、调用 push_back() 时向量中的迭代器无效吗?

当向 vector 调用 push_back 时,迭代器的有效性取决于是否发生了内存重新分配。如果 vector 当前的 capacity 足够容纳新元素,那么 push_back 不会导致迭代器失效。但如果 capacity 不足,vector 会分配一个更大的内存块(通常是当前大小的1.5或2倍),并将所有元素复制到新位置,这时之前的所有迭代器都会失效。

为了避免迭代器失效的问题,有以下几种方法:

  • 预先使用 reserve 分配足够的空间

  • 在 push_back 后重新获取迭代器

  • 使用索引而不是迭代器来遍历 vector

  • 如果必须使用迭代器,可以先完成迭代器操作,再进行 push_back

在实际编程中,建议在知道 vector 大致容量的情况下,先调用 reserve 预分配空间。这不仅能避免迭代器失效,还能提高程序性能,因为减少了内存重新分配的次数。

好了,面试题讲解就到这里,更多优质资源免费领取,欢迎大家扫码「CSDN 知识分享官」

ad1 webp
ad2 webp
ad1 webp
ad2 webp