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

C++高频知识点(六)

文章目录

  • 26. 说说malloc/free 和 new/delete区别
  • 27. 在C++程序中调用被C语言修饰的函数,为什么要加extern “C”
  • 28. 什么是内存泄漏?什么是野指针?什么是内存越界?如何避免?
  • 29. 内联函数有什么优点?内联函数和宏定义的区别
    • 内联函数的优点
    • 内联函数和宏定义的区别
    • 内联函数
    • 宏定义
    • 内联函数的类型检查和作用域
    • 宏定义的类型问题
  • 30. 什么时候要用虚析构函数
    • 没有虚析构函数的情况
    • 使用虚析构函数的情况

26. 说说malloc/free 和 new/delete区别

至少要知道标注绿色的这四项
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述

27. 在C++程序中调用被C语言修饰的函数,为什么要加extern “C”

在这里插入图片描述
示例:
假设你有一个C函数库,其中有一个函数void foo();。在C中,你可以直接在头文件中声明这个函数:

// C头文件:mylib.h  
void foo();

但是,如果你在C++程序中使用这个头文件,你需要告诉C++编译器这个函数是C风格的:

// C++源文件:main.cpp  
extern "C" {  
#include "mylib.h"  
}  int main() {  foo();  return 0;  
}

或者,你也可以在C头文件中使用extern “C”:

// 修改后的C头文件:mylib.h(兼容C++)  
#ifdef __cplusplus  
extern "C" {  
#endif  void foo();  #ifdef __cplusplus  
}  
#endif

这样,无论你的C++程序如何包含这个头文件,它都会知道foo()是一个C风格的函数。
在这里插入图片描述

28. 什么是内存泄漏?什么是野指针?什么是内存越界?如何避免?

在这里插入图片描述
在这里插入图片描述
在这里插入图片描述

裸指针(Raw Pointer)是 C++ 中最基础的指针类型,它直接指向内存地址,不提供任何智能功能或内存管理功能。裸指针是与 智能指针(如 std::unique_ptr、std::shared_ptr 等)相对的概念。

  1. 裸指针的定义
    裸指针就是没有附加任何智能功能的普通指针,它只是一个指向内存的地址,不会自动管理资源的生命周期。裸指针的最常见用途是指向动态分配的内存或者数组中的元素。
int* p = nullptr;  // 裸指针,初始化为空指针
  1. 裸指针的特点
  • 直接指向内存:裸指针直接存储对象的地址或内存位置。
  • 没有内存管理:裸指针不负责管理其所指向的内存(例如,自动释放内存)。这意味着程序员需要手动管理内存的分配和释放。
  • 不提供安全检查:裸指针不会检查访问无效内存或空指针的行为,导致程序容易发生悬挂指针、空指针解引用等错误。

悬挂指针:如果裸指针指向的内存已经被释放(例如,delete 后),但是指针仍然指向这个已释放的地址,就会造成 悬挂指针,这种指针是非法的,解引用它会导致未定义行为。

野指针和悬挂指针是指向无效内存地址的指针,通常它们是可以互换使用的术语,尤其在日常的讨论中。
在这里插入图片描述

29. 内联函数有什么优点?内联函数和宏定义的区别

内联函数(inline function)是C++中的一种优化机制,建议编译器在调用该函数时,将函数代码直接插入到调用点,而不是进行一般的函数调用。这样可以减少函数调用的开销,提高程序的执行效率

内联函数的优点和与宏定义的区别如下:

内联函数的优点

  1. 消除函数调用开销:内联函数通过将函数代码直接插入到调用点,避免了普通函数调用时的参数压栈、跳转和返回等开销。
  2. 增强代码可读性和可维护性:相比宏定义,内联函数具有函数的特性,支持类型检查和作用域规则,使代码更加安全和可读。
  3. 允许调试:内联函数可以在调试过程中正确地设置断点和查看堆栈信息,而宏定义展开的代码则无法直接调试。
  4. 支持多态性:内联函数可以是成员函数,支持类的继承和多态特性,而宏定义则不能。

内联函数和宏定义的区别

在这里插入图片描述

内联函数

#include <iostream>inline int add(int a, int b) {return a + b;
}int main() {std::cout << "Sum: " << add(3, 4) << std::endl; // 输出: Sum: 7return 0;
}

宏定义

#include <iostream>#define ADD(a, b) ((a) + (b))int main() {std::cout << "Sum: " << ADD(3, 4) << std::endl; // 输出: Sum: 7return 0;
}

注意:这里有两个额外的括号。这些括号在某些情况下是必要的,以确保宏的正确扩展和计算。否则,例如:你使用了像ADD(i++, j)这样的表达式,那么结果可能会与预期的不同。使用括号可以避免这种“运算符优先级”的问题

内联函数的类型检查和作用域

#include <iostream>inline int square(int x) {return x * x;
}int main() {std::cout << "Square: " << square(5) << std::endl; // 输出: Square: 25// std::cout << "Square: " << square("5") << std::endl; // 编译错误:类型不匹配return 0;
}

宏定义的类型问题

#include <iostream>#define SQUARE(x) ((x) * (x))int main() {std::cout << "Square: " << SQUARE(5) << std::endl; // 输出: Square: 25std::cout << "Square: " << SQUARE("5") << std::endl; // 运行时错误,文本替换后类型不匹配return 0;
}

30. 什么时候要用虚析构函数

在这里插入图片描述

没有虚析构函数的情况

#include <iostream>class Base {
public:Base() { std::cout << "Base constructor\n"; }~Base() { std::cout << "Base destructor\n"; }
};class Derived : public Base {
public:Derived() { std::cout << "Derived constructor\n"; }~Derived() { std::cout << "Derived destructor\n"; }
};int main() {Base* obj = new Derived();delete obj;  // 只调用了Base的析构函数,未调用Derived的析构函数return 0;
}

使用虚析构函数的情况

#include <iostream>class Base {
public:Base() { std::cout << "Base constructor\n"; }virtual ~Base() { std::cout << "Base destructor\n"; }
};class Derived : public Base {
public:Derived() { std::cout << "Derived constructor\n"; }~Derived() { std::cout << "Derived destructor\n"; }
};int main() {Base* obj = new Derived();delete obj;  // 调用了Base和Derived的析构函数return 0;
}

在这里插入图片描述

之后我会持续更新,如果喜欢我的文章,请记得一键三连哦,点赞关注收藏,你的每一个赞每一份关注每一次收藏都将是我前进路上的无限动力 !!!↖(▔▽▔)↗感谢支持!

http://www.xdnf.cn/news/14916.html

相关文章:

  • 【Python】Python Socket 编程详解:从原理到实践
  • 基于Java+Maven+Testng+Selenium+Log4j+Allure+Jenkins搭建一个WebUI自动化框架(1)搭建框架基本雏形
  • 实战:Android 15 (API 35) 适配 构建踩坑全记录
  • 鼎捷T100程序开发(R报表开发完整流程)
  • Bright Data MCP+Trae :快速构建电商导购助手垂直智能体
  • RabbitMQ第一章(MQ基础与RocketMQ使用手册)
  • EXCEL链接模板无法自动链接到PowerBI?试试这个方法
  • 数据分析-58-SPC统计过程控制的8个判异准则
  • 解决Maven“无法将类 XXXXX 中的构造器 XXXXXX 应用到给定类型”错误
  • WPA2 与 WPA3:深入解析Wi-Fi安全协议
  • 第0章:开篇词 - 嘿,别怕,AI应用开发没那么神!
  • Linux命令的命令历史
  • 安卓10.0系统修改定制化____recovery-from-boot.p文件的具体作用 在定制项目中的关联
  • stm32--SPI原理应用W25Q64(二)
  • 【力扣(LeetCode)】数据挖掘面试题0002:当面对实时数据流时您如何设计和实现机器学习模型?
  • vue openlayer 找出直线上的某一个点 , 点距离直线 最短路径的点 WKT坐标转换为GeoJSON坐标
  • xbox one controller DSLogic 逻辑分析仪截包
  • 商品中心—库存分桶高并发的优化文档
  • 谢飞机的Java高级开发面试:从Spring Boot到分布式架构的蜕变之旅
  • C++11标准库算法:深入理解std::none_of
  • Docker 将镜像打成压缩包将压缩包传到服务器运行
  • 创客匠人创始人IP打造实录:从行业观察者到生态构建者
  • Rust 所有权系统:深入浅出指南
  • MapReduce 学习
  • 【K线训练软件研发历程】【日常记录向】1.K线滑动窗口
  • DVWA | SQL Injection 数据库注入
  • Flutter 入门
  • ubuntu server系统 安装宝塔
  • 【算法训练营Day09】栈与队列part1
  • 内网使用rustdesk搭建远程桌面详细版