嵌入式开发高频面试题——第二章 C/C++高频面试题(下)

2024-08-30 21:36

本文主要是介绍嵌入式开发高频面试题——第二章 C/C++高频面试题(下),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

目录

      • 2.2 继承、多态相关面试题
        • 2.2.1 继承和虚继承⭐⭐⭐⭐⭐
        • 2.2.2 多态的类,内存布局是怎么样的 ⭐⭐⭐⭐⭐
        • 2.2.3 被隐藏的基类函数如何调用或者子类调用父类的同名函数和父类成员变量 ⭐⭐⭐⭐⭐
        • 2.2.4 多态实现的三个条件、实现的原理 ⭐⭐⭐⭐⭐
        • 2.2.5 对拷贝构造函数 深浅拷贝的理解 拷贝构造函数作用及用途?什么时候需要自定义拷贝构造函数? ⭐⭐⭐
        • 2.2.6 析构函数可以抛出异常吗?为什么不能抛出异常?除了资源泄露,还有其他需考虑的因素吗? ⭐⭐⭐
        • 2.2.7 什么情况下会调用拷贝构造函数(三种情况) ⭐⭐⭐
        • 2.2.8 析构函数一般写成虚函数的原因 ⭐⭐⭐⭐⭐
        • 2.2.9 构造函数为什么一般不定义为虚函数 ⭐⭐⭐⭐⭐
        • 2.2.10 什么是纯虚函数 ⭐⭐⭐⭐⭐
        • 2.2.11 静态绑定和动态绑定的介绍 ⭐⭐⭐⭐
        • 2.2.12 C++所有的构造函数 ⭐⭐⭐
        • 2.2.13 重写、重载、覆盖的区别 ⭐⭐⭐⭐⭐
        • 2.2.14 成员初始化列表的概念,为什么用成员初始化列表会快一些(性能优势)? ⭐⭐⭐⭐
        • 2.2.15 如何避免编译器进行的隐式类型转换;(`explicit`) ⭐⭐⭐⭐

2.2 继承、多态相关面试题

2.2.1 继承和虚继承⭐⭐⭐⭐⭐
  • 继承:继承是面向对象编程的核心特性之一。通过继承,一个类可以获取另一个类的所有属性和方法。继承分为公有继承、私有继承和保护继承:

    • 公有继承:基类的公有成员在派生类中仍然是公有的,基类的保护成员在派生类中是保护的。
    • 私有继承:基类的公有和保护成员在派生类中都变为私有的。
    • 保护继承:基类的公有和保护成员在派生类中都变为保护的。
  • 虚继承:虚继承用于解决多重继承时的“菱形继承问题”,即多个基类拥有同一个共同基类时,该共同基类在派生类中的副本问题。通过虚继承,共同基类在最派生类中只保留一份副本。

2.2.2 多态的类,内存布局是怎么样的 ⭐⭐⭐⭐⭐
  • 多态的类中包含虚函数时,编译器会为该类创建一个虚函数表(vtable),虚表中存储了类的虚函数的地址。
  • 每个对象会保存一个虚表指针(vptr),该指针指向类的虚函数表。
  • 当调用虚函数时,程序通过对象的虚表指针找到虚表,再根据虚表中保存的函数地址调用具体的函数。
  • 多态类的内存布局通常包含:
    • 非静态数据成员。
    • 一个隐藏的指向虚函数表的指针(vptr)。
2.2.3 被隐藏的基类函数如何调用或者子类调用父类的同名函数和父类成员变量 ⭐⭐⭐⭐⭐
  • 如果子类的函数隐藏了基类的同名函数,可以通过使用作用域解析运算符(::)来调用基类的函数。例如:

    class Base {
    public:void display() { std::cout << "Base class" << std::endl; }
    };class Derived : public Base {
    public:void display() { std::cout << "Derived class" << std::endl; }
    };int main() {Derived obj;obj.display();           // 调用子类的 displayobj.Base::display();     // 调用基类的 display
    }
    
  • 子类可以使用 Base::成员名 形式来访问基类中的同名变量或函数。

2.2.4 多态实现的三个条件、实现的原理 ⭐⭐⭐⭐⭐

多态的实现需要满足以下三个条件:

  1. 继承:必须存在继承关系,派生类继承基类。
  2. 虚函数:基类中的函数必须声明为虚函数(使用 virtual 关键字),以允许在运行时动态绑定。
  3. 基类指针或引用:必须通过基类的指针或引用来调用虚函数,以实现动态绑定。

多态的实现原理

  • 多态的核心是通过虚函数表(vtable)和虚表指针(vptr)来实现的。编译器为含有虚函数的类生成虚函数表,表中包含虚函数的地址。每个对象都有一个隐藏的虚表指针(vptr),该指针指向该类的虚表。通过基类指针调用虚函数时,程序会通过虚表指针找到对应的虚函数地址,从而实现动态绑定。
2.2.5 对拷贝构造函数 深浅拷贝的理解 拷贝构造函数作用及用途?什么时候需要自定义拷贝构造函数? ⭐⭐⭐
  • 拷贝构造函数:拷贝构造函数是用于创建新对象时,将已有对象的内容拷贝到新对象中。其形式为 ClassName(const ClassName &other)
  • 深拷贝:深拷贝会递归地复制对象中的指针指向的数据,确保新对象拥有独立的内存空间。
  • 浅拷贝:浅拷贝只复制对象的指针,因此两个对象共享相同的内存数据。若其中一个对象被销毁或释放内存,另一个对象会导致非法访问。

什么时候需要自定义拷贝构造函数

  • 当类包含动态分配内存(如指针、堆内存)时,通常需要自定义拷贝构造函数以避免浅拷贝导致内存共享或释放问题。
2.2.6 析构函数可以抛出异常吗?为什么不能抛出异常?除了资源泄露,还有其他需考虑的因素吗? ⭐⭐⭐
  • 析构函数不能抛出异常,因为在异常处理的过程中,如果析构函数再抛出异常,会导致程序终止(std::terminate())。
  • 除了资源泄露,还需要考虑到对象的销毁过程的完整性,以及如何安全地释放内存或其他系统资源。
2.2.7 什么情况下会调用拷贝构造函数(三种情况) ⭐⭐⭐

拷贝构造函数通常在以下三种情况下被调用:

  1. 当一个对象被作为参数按值传递给函数时。
  2. 当一个对象作为函数返回值按值返回时。
  3. 当一个对象用另一个对象初始化时(如 MyClass obj2 = obj1;)。
2.2.8 析构函数一般写成虚函数的原因 ⭐⭐⭐⭐⭐

析构函数通常被声明为虚函数,以确保在多态环境下,基类指针删除派生类对象时,可以正确调用派生类的析构函数,从而避免内存泄露和未释放的资源。

2.2.9 构造函数为什么一般不定义为虚函数 ⭐⭐⭐⭐⭐
  • 构造函数不应该是虚函数,因为在构造对象时,虚函数表还没有建立起来,所以无法调用虚函数。相反,析构函数是可以为虚函数的,因为对象被销毁时虚表已经存在,析构函数可以实现多态行为。
2.2.10 什么是纯虚函数 ⭐⭐⭐⭐⭐
  • 纯虚函数是一个不提供任何实现的虚函数,其声明形式为 virtual void func() = 0;
  • 纯虚函数用于创建抽象类,不能直接实例化,通常用于定义接口,派生类必须实现这些函数。
2.2.11 静态绑定和动态绑定的介绍 ⭐⭐⭐⭐
  • 静态绑定:函数调用在编译时就确定,编译器通过函数签名来决定调用哪个函数,适用于非虚函数。
  • 动态绑定:函数调用在运行时决定,基于虚函数表实现,用于虚函数。动态绑定是多态的基础。
2.2.12 C++所有的构造函数 ⭐⭐⭐

C++ 中的构造函数包括:

  1. 默认构造函数:没有参数的构造函数。
  2. 参数化构造函数:带有参数的构造函数。
  3. 拷贝构造函数:使用已有对象创建新对象的构造函数。
  4. 移动构造函数:接受右值引用作为参数的构造函数,避免不必要的拷贝。
2.2.13 重写、重载、覆盖的区别 ⭐⭐⭐⭐⭐
  • 重写(Override):子类重新定义基类的虚函数,必须与基类函数具有相同的函数签名。
  • 重载(Overload):在同一个作用域内,函数名相同但参数不同的多个函数定义。
  • 覆盖(Hide):子类定义了与基类同名的非虚函数时,基类的同名函数被隐藏。
2.2.14 成员初始化列表的概念,为什么用成员初始化列表会快一些(性能优势)? ⭐⭐⭐⭐
  • 成员初始化列表:用于在构造函数之前直接初始化类的数据成员,例如:MyClass(int a) : value(a) {}
  • 性能优势:使用初始化列表可以避免默认构造后再赋值,从而提高性能。尤其是当成员变量是常量、引用或没有默认构造函数时,必须使用初始化列表。
2.2.15 如何避免编译器进行的隐式类型转换;(explicit) ⭐⭐⭐⭐

使用 explicit 关键字来避免编译器执行隐式转换。例如,explicit MyClass(int a) 将防止从整数到类对象的隐式转换

这篇关于嵌入式开发高频面试题——第二章 C/C++高频面试题(下)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1121996

相关文章

从入门到精通C++11 <chrono> 库特性

《从入门到精通C++11<chrono>库特性》chrono库是C++11中一个非常强大和实用的库,它为时间处理提供了丰富的功能和类型安全的接口,通过本文的介绍,我们了解了chrono库的基本概念... 目录一、引言1.1 为什么需要<chrono>库1.2<chrono>库的基本概念二、时间段(Durat

C++20管道运算符的实现示例

《C++20管道运算符的实现示例》本文简要介绍C++20管道运算符的使用与实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧... 目录标准库的管道运算符使用自己实现类似的管道运算符我们不打算介绍太多,因为它实际属于c++20最为重要的

Visual Studio 2022 编译C++20代码的图文步骤

《VisualStudio2022编译C++20代码的图文步骤》在VisualStudio中启用C++20import功能,需设置语言标准为ISOC++20,开启扫描源查找模块依赖及实验性标... 默认创建Visual Studio桌面控制台项目代码包含C++20的import方法。右键项目的属性:

SQLite3 在嵌入式C环境中存储音频/视频文件的最优方案

《SQLite3在嵌入式C环境中存储音频/视频文件的最优方案》本文探讨了SQLite3在嵌入式C环境中存储音视频文件的优化方案,推荐采用文件路径存储结合元数据管理,兼顾效率与资源限制,小文件可使用B... 目录SQLite3 在嵌入式C环境中存储音频/视频文件的专业方案一、存储策略选择1. 直接存储 vs

嵌入式数据库SQLite 3配置使用讲解

《嵌入式数据库SQLite3配置使用讲解》本文强调嵌入式项目中SQLite3数据库的重要性,因其零配置、轻量级、跨平台及事务处理特性,可保障数据溯源与责任明确,详细讲解安装配置、基础语法及SQLit... 目录0、惨痛教训1、SQLite3环境配置(1)、下载安装SQLite库(2)、解压下载的文件(3)、

c++中的set容器介绍及操作大全

《c++中的set容器介绍及操作大全》:本文主要介绍c++中的set容器介绍及操作大全,本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友参考下吧... 目录​​一、核心特性​​️ ​​二、基本操作​​​​1. 初始化与赋值​​​​2. 增删查操作​​​​3. 遍历方

解析C++11 static_assert及与Boost库的关联从入门到精通

《解析C++11static_assert及与Boost库的关联从入门到精通》static_assert是C++中强大的编译时验证工具,它能够在编译阶段拦截不符合预期的类型或值,增强代码的健壮性,通... 目录一、背景知识:传统断言方法的局限性1.1 assert宏1.2 #error指令1.3 第三方解决

C++11委托构造函数和继承构造函数的实现

《C++11委托构造函数和继承构造函数的实现》C++引入了委托构造函数和继承构造函数这两个重要的特性,本文主要介绍了C++11委托构造函数和继承构造函数的实现,具有一定的参考价值,感兴趣的可以了解一下... 目录引言一、委托构造函数1.1 委托构造函数的定义与作用1.2 委托构造函数的语法1.3 委托构造函

C++11作用域枚举(Scoped Enums)的实现示例

《C++11作用域枚举(ScopedEnums)的实现示例》枚举类型是一种非常实用的工具,C++11标准引入了作用域枚举,也称为强类型枚举,本文主要介绍了C++11作用域枚举(ScopedEnums... 目录一、引言二、传统枚举类型的局限性2.1 命名空间污染2.2 整型提升问题2.3 类型转换问题三、C

C++链表的虚拟头节点实现细节及注意事项

《C++链表的虚拟头节点实现细节及注意事项》虚拟头节点是链表操作中极为实用的设计技巧,它通过在链表真实头部前添加一个特殊节点,有效简化边界条件处理,:本文主要介绍C++链表的虚拟头节点实现细节及注... 目录C++链表虚拟头节点(Dummy Head)一、虚拟头节点的本质与核心作用1. 定义2. 核心价值二