C++反汇编,指针和内存分配细节,面试题05

server/2024/10/21 18:31:25/

文章目录

  • 20. 指针 vs 引用
  • 21. new vs malloc

20. 指针 vs 引用

  • 指针是实体,占用内存空间,逻辑上独立;引用是别名,与变量共享内存空间,逻辑上不独立。
  • 指针定义时可以不初始化;引用定义时必须初始化。
  • 指针的值可以变,即可以指向别的内存地址;引用不变的。
  • 指针可以为nullptr;引用不能为空。
  • sizeof(指针)计算指针大小,由于指针保存的是内存地址,所以无论什么类型的指针,在32位程序里占4B,在64位程序里占8B;而sizeof(引用)计算引用对象的大小。
  • 指针自增用于地址偏移,运算结果取决于指针类型,因为当指针保存的是数组首地址时,为了能够利用指针自增后访问到数组下一成员,所以加的是类型长度,而非1,如下图;引用自增是引用对象自增。
    在这里插入图片描述
  • 指针取出数据内容时需要解引用;引用不需要。
  • 有二级指针;没有二级引用。
  • 【注意】 如果返回动态分配的内存,必须使用指针,使用引用会内存泄漏,如下图。
    在这里插入图片描述
    分析1:使用CRT库检测内存泄漏,4B,因为不能删除引用指向的内存地址,即使用 delete& ref 会触发中断
    在这里插入图片描述
    分析2:相比之下,使用指针返回动态分配的内存,可以在不需要时使用 delete释放内存,从而避免内存泄漏。

21. new vs malloc

  • new是C++运算符;malloc是C的库函数。
  • new返回指定类型指针;malloc返回void*指针,需要强制类型转换。
  • new自动计算需分配的空间;malloc需要指定分配空间的大小。
  • new可以被重载;malloc不能,代码如下。
#include<iostream>
using namespace std;// 重载 new 操作符
void* operator new(size_t size) 
{cout << "Custom new: Allocating " << size << " bytes" << endl;void* ptr = std::malloc(size);if (!ptr) {throw std::bad_alloc(); // 内存分配失败时抛出异常}return ptr;
}// 重载 delete 操作符
void operator delete(void* ptr) noexcept {cout << "Custom delete: Freeing memory" << endl;free(ptr);
}int main() 
{int* arr = new int(4); delete arr;return 0;
}

程序运行结果,如下图。
在这里插入图片描述

  • 代码如下,进行以下分析。
#define _CRTDBG_MAP_ALLOC
#include <stdlib.h>
#include <crtdbg.h>
#include<iostream>
using namespace std;class MyClass {
private:int* value;
public:MyClass(int v) : value(new int(v)) {printf("%s\n", "Constructor class");}~MyClass(){delete value;printf("%s\n", "Delete class");}
};int main() 
{MyClass* obj1 = new MyClass(42);MyClass* obj2 = (MyClass*)malloc(sizeof(MyClass));new(obj2) MyClass(44);delete obj1;free(obj2);_CrtDumpMemoryLeaks();return 0;
}
  • new会调用构造函数;malloc不会,即malloc只是分配内存空间,需要在其他地方初始化,如下图。
    在这里插入图片描述
    分析1:MyClass类大小,看它的数据成员大小,即4B。使用new时,先申请4B内存空间,然后调用构造函数进行初始化。
    请添加图片描述
    分析2:使用malloc时,只申请4B内存空间。然后使用new(obj2) MyClass(44);显示调用构造函数,进行初始化。
    在这里插入图片描述
    分析3:new的内部实现,会调用malloc分配内存空间。

  • 【注意】 new使用delete释放内存空间,在释放前会调用析构函数;malloc使用free释放内存空间,由于不会调用析构函数,会造成内存泄漏,如下图。
    请添加图片描述
    在这里插入图片描述

分析1:使用delete时,先调用析构函数释放对象内存空间,再释放指针内存空间。
请添加图片描述
分析2:使用free时,只释放指针内存空间,由于对象内存空间无法释放,造成内存泄漏。


http://www.ppmy.cn/server/38922.html

相关文章

zookeeper启动后占用8080端口问题分析及解决

ZooKeeper是一个分布式的&#xff0c;开放源码的分布式应用程序协调服务。它为分布式应用提供一致性服务的软件&#xff0c;提供的功能包括&#xff1a;配置维护、域名服务、分布式同步、组服务等。 我们经常在运行zookeeper服务时&#xff0c;不需要配置服务端口&#xff0c;…

接口测试及常用的接口测试工具(Postman/Jmeter)

&#x1f345; 视频学习&#xff1a;文末有免费的配套视频可观看 &#x1f345; 点击文末小卡片 &#xff0c;免费获取软件测试全套资料&#xff0c;资料在手&#xff0c;涨薪更快 首先&#xff0c;什么是接口呢&#xff1f; 接口一般来说有两种&#xff0c;一种是程序内部的接…

React 之 useMemo Hook (九)

useMemo 是 React 的一个Hook&#xff0c;它允许你“记住”一些计算值&#xff0c;只有在依赖项之一发生变化时才会重新计算这些值。这有助于避免不必要的重新计算和渲染&#xff0c;从而提高应用程序的性能。 代码栗子&#xff08;计算一个斐波那契数列的值&#xff09;&#…

普通组件的注册-局部注册和全局注册

目录 一、局部注册和全局注册-概述 二、局部注册的使用示例 三、全局注册的使用示例 一、局部注册和全局注册-概述 组件注册有两种方式&#xff1a; 局部注册&#xff1a;只能在注册的组件内使用。使用方法&#xff1a;创建.vue文件&#xff0c;在使用的组件内导入并注册。…

Android BINDER是干嘛的?

1.系统架构 2.binder 源码位置&#xff1a; 与LINUX传统IPC对比

Redis-5 分布式锁

一.为什么要使用分布式锁&#xff1f; 传统的互斥锁synchronized只能作用于同一台虚拟机上的线程&#xff0c;在使用服务器集群部署的情况下&#xff0c;互斥锁就会失效&#xff0c;因此要采用分布式锁来处理不同服务器上的线程访问同一资源的情况。 二.redis的分布式锁是如何…

【Qt】深入理解QWidget常用控件: enable属性、geometry属性和window frame属性

文章目录 前言&#xff1a;1. 什么是控件2. Qt中QWidget控件的常用属性及元编程QWidget 核心属性enable属性&#xff1a;geometry 属性 :window frame 窗口框架 总结: 前言&#xff1a; 图形化界面的开发常常需要使用各种控件&#xff0c;而Qt作为一个强大的跨平台GUI应用程序…

Java实现NFS文件上传、下载和读取功能的工具类

Java实现NFS文件上传、下载和读取功能的工具类 引言&#xff1a;代码示例一、准备工作二、工具类设计与核心方法三、异常处理与性能优化四、总结 引言&#xff1a; NFS&#xff08;Network File System&#xff09;广泛应用于分布式环境的情况下&#xff0c;这里介绍使用Java工…