【C++ 面试突击 · 05】大厂高频面试题:从内联函数到内存管理全梳理
目录一、什么是inline函数二、inline函数的优缺点三、inline和宏定义的比较四、虚函数virtual可以是内联函数inline吗五、C中struct和class的区别六、面向对象三大特征七、静态多态和动态多态的区别八、什么是虚析构函数九、虚析构函数的作用十、malloc、calloc、realloc、alloca的区别和用法一、什么是inline函数inline内联函数是C中用于建议编译器进行内联展开的函数。它通过在编译时将函数体直接插入到调用处避免函数调用的开销如压栈、跳转、返回等从而提升程序运行效率。核心特点编译期展开函数体在调用处直接展开无函数调用开销。代码膨胀风险若函数体过大或调用频繁可能导致代码体积显著增加。编译器决定权inline仅为建议编译器可根据实际情况如函数复杂度、调用频率决定是否真正内联。二、inline函数的优缺点优点性能提升避免函数调用的压栈、跳转、返回等开销适合频繁调用的小函数。代码可读性可将常用的小功能封装为函数同时保持性能接近宏定义。缺点代码膨胀函数体被多次展开可能导致可执行文件体积增大。调试困难内联后函数调用栈被破坏调试时难以定位具体调用位置。编译器限制复杂函数如递归、循环、大函数体可能无法内联。三、inline和宏定义的比较特性inline函数宏定义#define类型安全支持类型检查参数类型明确无类型检查参数替换可能导致类型错误调试支持支持调试可设置断点无调试支持展开后代码难以追踪作用域遵循C作用域规则无作用域全局替换编译器优化编译器可进行优化如内联、常量折叠仅文本替换无编译器优化适用场景小函数、频繁调用的函数简单的常量定义、宏函数需谨慎使用四、虚函数virtual可以是内联函数inline吗可以但需注意以下几点内联展开条件虚函数的内联展开仅在编译时能确定调用对象的情况下有效如通过对象调用而非指针/引用。动态绑定限制若通过指针或引用调用虚函数由于动态绑定运行时决定调用哪个函数编译器无法确定具体函数体因此无法内联。性能权衡虚函数的动态绑定开销通常大于内联带来的性能提升因此虚函数内联的实际意义有限。五、C中struct和class的区别特性structclass默认访问权限publicprivate默认继承方式publicprivate使用场景通常用于数据结构如POD类型通常用于面向对象编程封装、继承、多态功能支持成员函数、继承、多态与class功能一致支持所有C特性封装、继承、多态核心区别struct更偏向于数据结构class更偏向于面向对象编程但两者在功能上几乎完全一致仅默认访问权限和继承方式不同。六、面向对象三大特征封装Encapsulation将数据属性和操作数据的方法行为绑定在一起对外隐藏内部实现细节仅暴露必要的接口。通过访问控制private、protected、public实现。继承Inheritance子类继承父类的属性和方法实现代码复用。支持单继承、多继承C支持可扩展父类功能。多态Polymorphism同一接口在不同对象中表现出不同行为。分为静态多态编译时多态如函数重载、模板和动态多态运行时多态如虚函数、接口实现。七、静态多态和动态多态的区别特性静态多态编译时多态动态多态运行时多态实现方式函数重载、模板、运算符重载虚函数、接口实现绑定时间编译时确定调用哪个函数运行时通过虚函数表vtable动态绑定性能开销无运行时开销效率高有虚函数调用开销查表、跳转灵活性编译时确定灵活性较低运行时确定支持多态扩展八、什么是虚析构函数虚析构函数是基类中声明为virtual的析构函数。当通过基类指针删除派生类对象时虚析构函数确保派生类的析构函数先被调用再调用基类的析构函数避免资源泄漏。示例class Base { public: virtual ~Base() { /* 基类析构逻辑 */ } }; class Derived : public Base { public: ~Derived() { /* 派生类析构逻辑如释放资源 */ } }; Base* ptr new Derived(); delete ptr; // 虚析构确保先调用~Derived()再调用~Base()九、虚析构函数的作用确保正确析构通过基类指针删除派生类对象时避免仅调用基类析构函数导致派生类资源未释放如内存泄漏、文件句柄未关闭。支持多态删除在多态场景中如容器存储基类指针虚析构函数保证所有派生类对象被正确析构。避免未定义行为若基类析构函数非虚删除派生类对象时可能引发未定义行为如资源泄漏、程序崩溃。十、malloc、calloc、realloc、alloca的区别和用法函数功能内存位置初始化释放方式适用场景malloc(size_t size)分配指定大小的内存块堆区未初始化内容随机free(ptr)需要手动初始化的内存分配calloc(size_t nmemb, size_t size)分配nmemb个元素每个size字节初始化为0堆区初始化为0free(ptr)需要初始化为0的内存分配如数组realloc(void* ptr, size_t size)调整已分配内存块的大小可扩展或收缩堆区内容保留新内存未初始化free(ptr)动态调整内存大小如数组扩容alloca(size_t size)在栈区分配内存栈区未初始化无需手动释放函数返回时自动释放临时小内存分配避免堆分配开销关键区别内存位置malloc/calloc/realloc分配在堆区需手动freealloca分配在栈区函数返回时自动释放。初始化calloc自动初始化为0malloc和realloc不初始化。安全性alloca在栈上分配若分配过大可能导致栈溢出realloc可能移动内存块需注意指针失效问题。
本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2460359.html
如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!