当前位置: 首页 > ds >正文

[C++面试] 基础题

1、类的特殊成员函数的总结(3-2-1)

  • 构造函数(三个构造)
    • 普通构造函数:用于创建对象并初始化其成员。
    • 拷贝构造函数:用已存在的对象创建新对象,进行深拷贝(若有资源管理时需自定义)。
    • 移动构造函数:以右值引用为参数,转移资源所有权(如指针),避免不必要的深拷贝,提高效率。
  • 赋值函数(两个赋值)
    • 普通赋值函数:实现对象间的赋值操作(需处理自赋值、深拷贝等情况)。
    • 移动赋值函数:通过右值引用转移资源,避免深拷贝开销,通常配合转移构造函数实现资源管理优化。
  • 析构函数(一个析构):用于对象生命周期结束时释放资源(如动态分配的内存、文件句柄等),确保资源不泄漏。

2、 移动赋值函数为什么不加const

拷贝赋值函数 MyString& MyString::operator=(const MyString& ob) 参数加 const,是为了:

  • 防止意外修改原对象:拷贝赋值仅复制原对象数据,不希望修改原对象,const 可避免函数内误操作修改原对象。
  • 提高通用性:能接受常量对象作为参数,若不加 const,则无法对常量对象进行拷贝赋值操作。

移动赋值函数 MyString& operator=(MyString&& ob) 不加 const:

  • 移动语义的核心是转移资源所有权,需要修改被移动对象(如将其内部指针置空,避免后续重复释放资源)。
  • 若参数加 const,则无法修改被移动对象,导致移动语义无法实现。

3、重载与覆盖

重载(Overloading)​

发生在同一作用域内(如同一个类中),允许存在多个同名函数,但参数列表必须不同​(参数类型、个数或顺序不同)

class Example {
public:void func(int a) { /*...*/ }       // 重载版本1void func(double b) { /*...*/ }    // 重载版本2(参数类型不同)void func(int a, int b) { /*...*/ } // 重载版本3(参数个数不同)
};
  • 返回值类型无关​:仅返回值不同不能构成重载
  • 编译时静态多态​:编译器根据调用时的实参类型和数量选择匹配的函数
  • 无虚函数要求​:普通成员函数即可重载

覆盖(Overriding)

发生在继承关系中​(派生类与基类之间),派生类重新定义基类的虚函数,要求函数签名(名称、参数列表、返回类型)完全一致

class Base {
public:virtual void show() { /*...*/ }  // 基类虚函数
};class Derived : public Base {
public:void show() override { /*...*/ } // 覆盖基类虚函数(C++11后推荐使用override关键字)
};
  • 必须虚函数​:基类函数需声明为virtual,否则派生类同名函数为隐藏而非覆盖
  • 运行时多态​:通过基类指针或引用调用时,根据对象实际类型决定执行哪个版本
  • 协变返回类型​:派生类返回类型可以是基类返回类型的派生类

当派生类重写基类的虚函数时,允许返回基类方法返回类型的派生类对象。​

  • Liskov替换原则​:派生类对象可替代基类对象使用
  • 隐式类型转换​:Dog* 可隐式转换为 Animal*,但反向转换需显式处理
// 基类
class Animal {
public:virtual Animal* clone() const {  // 返回基类指针return new Animal(*this);}virtual void speak() const { std::cout << "Animal sound" << std::endl; }
};// 派生类
class Dog : public Animal {
public:// 协变返回类型:返回派生类指针Dog* clone() const override {   return new Dog(*this);}void speak() const override { std::cout << "Bark!" << std::endl; }
};int main() {Animal* animal1 = new Dog();Animal* clonedAnimal = animal1->clone();  // 实际调用Dog::clone()clonedAnimal->speak();  // 输出 "Bark!"(无需类型转换)delete animal1;delete clonedAnimal;return 0;
}

若派生类定义了与基类同名但参数不同的函数(无论基类是否为虚函数),基类函数会被隐藏,而非覆盖

class Base { public: void func(int a) {} };
class Derived : public Base {void func(double b) {} // 隐藏基类的func(int),而非覆盖
};

C++11引入override关键字,强制检查派生类函数是否真正覆盖基类虚函数,避免因参数不匹配导致的隐藏错误 

4、C++支持函数重载而C语言不支持的原因

C++通过编译器对函数名进行"名称修饰(Name Mangling)"实现重载:

  • 根据函数参数类型、个数、顺序(容易忽略)生成唯一修饰名
  • 调用时通过符号表匹配正确函数实现
  • C语言不做名称修饰,要求函数名全局唯一,因此无法区分同名不同参的函数

5、函数重载注意事项

  • 必须满足参数列表差异(类型/个数/顺序)
  • 返回值类型不能作为重载依据(会导致编译错误)
  • 同名函数在不同作用域不构成重载
  • 注意隐式类型转换可能引发的二义性:
void func(int a) {}
void func(double b) {}
func(5);     // 明确调用int版
func(5.0);   // 明确调用double版
func('c');   // 调用int版(char→int隐式转换)

6、C/C++代码互调用规范

extern "C" { // C++ 调用C#include "c_header.h"
}
// cpp_code.cpp
extern "C" void cpp_func() { /*...*/ }

 在头文件中通过条件编译实现跨语言兼容:

#ifdef __cplusplus
extern "C" {
#endifvoid shared_function();#ifdef __cplusplus
}
#endif
http://www.xdnf.cn/news/8245.html

相关文章:

  • springboot 集成kerberos 用户认证 获取域账号
  • 完整改进RIME算法,基于修正多项式微分学习算子Rime-ice增长优化器,完整MATLAB代码获取
  • 【数据结构】 栈和队列
  • 微软全新开源的Agentic Web网络项目:NLWeb,到底是什么 ?
  • 鸿蒙App开发学习路径
  • JAVA|后端编码规范
  • 仿腾讯会议——视频发送接收
  • 计算机发展史
  • 从零基础到最佳实践:Vue.js 系列(7/10):《常用内置 API 与插件》
  • scratch课后一练--事件模块
  • Linux系统编程 | IPC对象---消息队列
  • DeepSeek:开启IT领域人效管理新时代
  • Java-根据路径获取JSON字符串的value值
  • zabbix 常见问题
  • 深入解析JVM垃圾回收器:原理、实践与调优指南
  • 实用重复文件批量处理工具
  • 关于SQL SERVER中round函数的用法和示例
  • 一台机器怎么部署k8s集群
  • React-fiber架构
  • Python可视化设计原则
  • 【424. 替换后的最长重复字符】
  • docker-compose常用命令介绍
  • 已经 上线 Vue 项目 国际化 i18n 中译英
  • OpenCV 图像对象的创建与赋值
  • Apollo10.0学习——planning模块(9)之参数详解一
  • Vscode +Keil Assistant编译报错处理
  • C++ -- vector
  • 系统性能分析基本概念(5) : 何时开始性能分析
  • 【语法】C++的map/set
  • 平安健康2025年一季度深耕医养,科技赋能见成效