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

iOS开发笔记之九十八——关于Memory Leak总结笔记

*****阅读完此文,大概需要3分钟******

关于Memory leak(内存泄漏)的问题,如果是面试被问这个问题以及此类问题,主要涉及下面3个方面:

  • 内存泄漏的常见场景有哪些,列举几个常见的例子?

  • 开发中实际发生了内存泄漏,该如何定位与解决?

开发中该如何避免内存泄漏,有哪些方法?

下面我们就分别从这个3个方面进行总结。

Memory Leak的常见场景

1. 循环引用

ARC机制下,循环引用是导致内存泄漏的一个主要原因,它也分为很多具体的场景,例如:

  • 两个类对象之间的循环引用;

  • Block中的循环引用问题(包括GCD或其他系统Block等);

  • Timer的循环引用问题;

  • delegate带来的循环引用问题;

属性观察、监听类等场景;

2. 对象生命周期管理(持有)不当

这种需要结合实际业务问题来分析,例如,

  • Static字段或其他全局对象导致的持有不当引起内存泄漏

集合类等的引用关系不当或未及时释放等。

3. 对象的未及时释放

有些对象需要手动释放内存,例如 Core Foundation(CF)和 Core Graphics(CG)框架的对象。如果不及时释放这些对象,就可能导致内存泄漏。

4. 资源的未及时关闭或释放

资源类未及时关闭(或未释放)也是一种常见的内存泄漏场景,例如:

  • 文件的打开后(FileHandle),未及时Close。

  • 网络请求的URLSession对象未及时释放。

Memory Leak该如何定位与解决

其实找到问题点后,基本很好解决,例如:

循环引用就想办法利用weak来打破循环;对象的生命周期不当可以更换持有者或者需要重新设计代码;CF或CG类的对象需要手动释放;资源类的占用也需要及时手动释放等等。

但是问题主要难在怎么找到泄漏代码。

1. 手动调试法

当我们意识到发生泄漏时,一般都会有明显的表象,例如VC Pop失败等。

如果我们对代码很熟悉,一般情况下,我相信很多人会直接找代码进行手动调试。常见手段有:

  • 重写dealloc或者deinit方法

我们可以尝试在一些类的dealloc或者deinit方法中断点或者加log,没有预期的执行,则为嫌疑泄漏点。

  • 手动检查引用计数

手动检查对象的引用计数,可以帮助我们找到是否存在引用计数错误导致的内存泄漏。我们可以使用 Objective-C 或 Swift 中的 retainCount 属性来手动检查对象的引用计数。

  • 写一些辅助测试代码

如果内存泄漏点比较隐晦,可以通过写一些测试用例的代码来复现,例如连续循环9999次,把问题扩大,引发内存耗尽崩溃的Crash,再根据堆栈定位代码,这也是定位概率性Crash的常用手段;

  • 逐一排查法

逐一注释掉问题的代码,直到发现问题点所在。开发中排查Crash问题,也常用类似手段,虽然看似很Low,但是十分有效。

2. 工具类定位

这种有很多可以借助,例如:

Instruments工具箱,其中的Leaks工具或者内存调试器Analyze来分析;

Xcode Memory Debugger中的Memory Graph,

可以帮助我们分析对象的内存引用关系,并找到循环引用和内存泄漏。

  • 第三方的检测工具,

例如MLeaksFinder、FBRetainCycleDetector、LeakDetector、HeapInspector等。

  • 静态分析工具,

例如 Clang 静态分析器、Infer、OCLint、SwiftLint等工具。

这些工具一般是附带性发现一些内存泄漏的代码,可以给我们一些提示或者警告。

如何避免Memory Leak?

实际开发中,不大可能,写段代码就跑一下Instruments这些工具,就连一些MLeaksFinder这些经常误报的工具都嫌烦会直接关闭。

那我们该如何避免,尽可能降低问题代码的产生?

第一,要有一些风险意识。例如Timer既然选择它,就该知道它的最大的风险就是容易导致内存问题。将一些容易导致泄漏的场景烂熟于心。还例如,谨慎使用全局变量或者单例等。

第二,要养成一些习惯。例如,写完一段代码或者接收一段代码,在dealloc或deinit中断点,保证对象可预期释放。还例如Swift中尽可能采取值类型,而非引用类型;开发完自测时,把一些内存检测工具都打开作为辅助检测;

第三,做好系统设计。有些业务场景很容易产生泄漏,例如曾经我参与过一个直播业务的开发,房间VC与房间的唯一数据模型对象dataModel之间很容易相互引用,由于dataModel承载了很多业务基础信息,业务对这个dataModel都极其依赖,这个对象也因此被传的很深很广,一个不小心就发生循环引用。针对这种开始的设计不当,后面只能打补丁修复(维护一个dataModel弱引用集合,每次用时根据id去查询并获得对应房间的dataModel对象,业务之间只需维护一个id的String对象或者存储最少信息的字典对象即可)。类似的业务场景还有商品详情页等。

第四,定时给工程代码做“体检”,一般工程量较大时,这个十分有必要,跑一下Instruments或Memory Graph,几乎每次都有收获。

欢迎关注公众号ios_hunter,不要错过更多优质文章更新。

相关文章:

iOS开发笔记之九十八——关于Memory Leak总结笔记

*****阅读完此文,大概需要3分钟******关于Memory leak(内存泄漏)的问题,如果是面试被问这个问题以及此类问题,主要涉及下面3个方面:内存泄漏的常见场景有哪些,列举几个常见的例子?开…...

HTML基础语法

一 前端简介构成语言说明结构HTML页面元素和内容表现CSS网页元素的外观和位置等页面样式(美化)行为JavaScript网页模型的定义和页面交互二 HTML1.简介HTML(Hyper Text Markup Language):超文本标记语言。网页结构整体&…...

微软新版必应gpt人工智能体验教程

大家好,我是雄雄,欢迎关注微信公众号:** 雄雄的小课堂 ** 现在是:2023年2月28日18:35:02 前言 前几天,发了一篇文章,主要介绍了如何申请新必应的内测名单,其实一共也就那几步,然后等着就行: 文章连接:new bing如何快速申请内测资格,从而体验人工智能? 今天,终于…...

你问我答|虚拟机、容器和无服务器,怎么选?

在新技术层出不穷的当下,每家企业都希望不断降低成本,并提高运营效率,一个方法就是寻找不同的技术方案来优化运营。      例如,曾经一台服务器只能运行一个应用(裸机);接着,一台服务器的资源可以划分为多个块,从而运行多个应用(虚拟化);再到后来,应用越来越多,为了方便它们…...

某建筑设计研究院“综合布线管理软件”应用实践

某建筑设计研究院有限公司(简称“某院”)隶属于国务院国资委直属的大型骨干科技型中央企业。“某院”前身为中央直属设计公司,创建于1952年。成立近70年来,始终秉承优良传统,致力于推进国内勘察设计产业的创新发展&…...

R语言绘制SCI论文中常见的箱线散点图,并自动进行方差分析计算显著性水平

显著性标记箱线散点图 本篇笔记的内容是在R语言中利用ggplot2,ggsignif,ggsci,ggpubr等包制作箱线散点图,并计算指定变量之间的显著性水平,对不同分组进行特异性标记,最终效果如下。 加载R包 library(ggplo…...

redux-saga

redux-saga 官网:About | Redux-Saga 中文网:自述 Redux-Saga redux-saga 是一个用于管理 异步获取数据(副作用) 的redux中间件;它的目标是让副作用管理更容易,执行更高效,测试更简单,处理故障时更容易… …...

【C++】-- 智能指针

目录 智能指针意义 智能指针的使用及原理 RAII 智能指针的原理 std::auto_ptr std::auto_ptr的模拟实现 std::unique_ptr std::unique_ptr模拟实现 std::shared_ptr std::shared_ptr的模拟实现 循环引用问题 智能指针意义 #问:为什么需要智能指针&#…...

数据结构与算法——4时间复杂度分析2(常见的大O阶)

这篇文章是时间复杂度分析的第二篇。在前一篇文章中,我们从0推导出了为什么要用时间复杂度,时间复杂度如何分析以及时间复杂度的表示三部分内容。这篇文章,是对一些常用的时间复杂度进行一个总结,相当于是一个小结论 1.常见的大O…...

IIS解析漏洞

IIS 6.0在解析文件时存在以下两个解析漏洞。 ①当建立*.asa、*.asp格式的文件夹时,其目录下的任意文件都将被IIS当作asp文件来解析。 例如:建立文件夹 parsing.asp,在 parsing.asp 文件夹内新建一个文本文档 test.txt,其内容为&…...

2023 年腾讯云轻量和CVM服务器租用价格表出炉(CPU/内存/带宽/系统盘)

腾讯云服务器的价格表是用户比较关心的问题,服务器的价格组成包括云服务器的机型价格、磁盘价格和宽带价格,主机教程网来详细说下腾讯云最新的云服务器价格表。我们以北京一区、Linux系统的云服务器为例,其他地域的价格会有所差异&#xff0c…...

Java学习之路002——面向对象编程

【说明】部分内容来源于网络,如有冲突,请联系作者删除。 一、面向对象编程(OOP) 2.1 对象和类的关系 2.2 面向对象的特征 2.2.1 封装 2.2.2 继承 2.2.3 多态 3、抽象 使用abstract关键字修饰的类或者方法 定义抽象类(使用abstract) // 1、定义抽象方法…...

VR直播丨颠覆性技术革命,新型直播已经到来

细数当下最火热的营销手段,首先浮现脑海的无疑是“直播”。前有罗永浩、李佳琦,后有刘畊宏和东方甄选,直播如日中天,俨然成了大众足不出户就能休闲娱乐的重要途径。 而随着虚拟现实在“十四五规划”中被列入“建设数字中国”数字…...

【微信小程序】-- WXSS 模板样式- rpx import (十三)

💌 所属专栏:【微信小程序开发教程】 😀 作  者:我是夜阑的狗🐶 🚀 个人简介:一个正在努力学技术的CV工程师,专注基础和实战分享 ,欢迎咨询! &…...

Biotin-PEG-SVA,生物素聚乙二醇琥珀酰亚胺戊酸酯,可用于检测或分子标记

Biotin-PEG-SVA 结构式:PEG分子量: 1000,2000,3400,5000,10000中文名称:生物素聚乙二醇琥珀酰亚胺戊酸酯,生物素-PEG-琥珀酰亚胺戊酸酯英文名称:Biotin-PEG-SVA &#xf…...

云原生是什么?核心概念和应用方法解析

什么是云原生? 云原生是一种基于容器、微服务和自动化运维的软件开发和部署方法。它可以使应用程序更加高效、可靠和可扩展,适用于各种不同的云平台。 如果要更直接通俗的来解释下上面的概念。云原生更准确来说就是一种文化,是一种潮流&…...

Editor工具开发实用篇:EditorGUI/EditorGUILayout的区别和EditorGUILayout的方法介绍

目录 一:EditorGUI和EditorGUILayout区别 二:EditorGUILayout 1.EditorGUILayout.BeginFadeGroup(float value); 2.EditorGUILayout.BeginHorizontal EditorGUILayout.BeginVertical 3.EditorGUILayout.BeginScrollView 4.EditorGUILayout.BeginT…...

(五十二)大白话不断在表中插入数据时,物理存储是如何进行页分裂的?.md

上回我们讲到了数据页的物理存储结构,数据页之间是组成双向链表的,数据页内部的数据行是组成单向链表的,每个数据页内根据主键做了一个页目录 然后一般来说,你没有索引的情况下,所有的数据查询,其实在物理…...

Unity 渲染顺序

Unity中的渲染顺序自上而下大致分为三层渲染优先级 Camera depth > Sorting Layer > Order in Layer > RenderQueueCamera depth:越小越优先(大的显示在小的前面)如图:尽管Sphere距离摄像机较远,但由于Camera_Sphere dep…...

短视频美颜sdk人脸编辑技术详解、美颜sdk代码分析

短视频美颜sdk中人脸编辑技术可以将人像风格进行转变,小编认为这也是未来的美颜sdk的一个重要发展方向,下文小编将为大家讲解一下短视频美颜sdk中人脸编辑的关键点。 一、人脸编辑的细分关键点 1、年龄 通过更改人脸的年龄属性,可用于模仿人…...

Flexible H-Tree实战:如何在复杂SoC设计中实现低延迟时钟分布(附Cadence Innovus配置指南)

Flexible H-Tree实战:复杂SoC设计中的低延迟时钟分布艺术 时钟网络就像芯片的神经系统,每一个脉冲都决定着数十亿晶体管的协同工作。在28nm以下的复杂SoC设计中,时钟分布网络的设计难度呈指数级增长——宏单元的不规则分布、跨电压域时序收敛…...

【Isaac Sim 4.5.0】从安装到启动:Ubuntu环境下的疑难杂症排查与修复实录

1. 环境准备:从零搭建Isaac Sim的硬件与软件基础 在Ubuntu系统上部署Isaac Sim之前,硬件兼容性检查是避免后续问题的关键第一步。我的RTX 4090显卡在安装过程中就遇到了显存识别问题,后来发现是PCIe供电不足导致的性能降频。建议先用lspci -…...

LyricsX完整指南:让桌面歌词显示更智能的Mac工具

LyricsX完整指南:让桌面歌词显示更智能的Mac工具 【免费下载链接】Lyrics Swift-based iTunes plug-in to display lyrics on the desktop. 项目地址: https://gitcode.com/gh_mirrors/lyr/Lyrics LyricsX是一款基于Swift开发的iTunes插件,专为Ma…...

跨品牌路由器桥接实战:TP-LINK(AC1200)与FAST(FWR303)混合组网方案

1. 为什么需要跨品牌路由器桥接? 家里WiFi信号差是很多人的痛点。我去年搬进新家时就遇到这个问题——书房和卫生间经常只有一格信号,视频通话卡成PPT。后来发现是承重墙太多,单一路由器根本穿不透。换更贵的路由器?成本太高。拉…...

用STM32F103和TMC2209给步进电机加个‘防丢步’外挂:手把手实现位置式PID闭环

用STM32F103和TMC2209给步进电机加个‘防丢步’外挂:手把手实现位置式PID闭环 步进电机在3D打印机、CNC机床和自动化设备中无处不在,但许多开发者都遇到过这样的尴尬:明明发送了1000个脉冲,电机却只转了980步。这种"丢步&quo…...

verl分布式训练实战:从单机多卡到多机多卡的完整配置指南

1. 分布式训练基础概念与verl框架简介 第一次接触分布式训练的朋友可能会被"单机多卡"、"多机多卡"这些术语吓到。其实理解起来很简单,就像搬家时找帮手一样:单机多卡相当于在一套房子里叫来几个家人一起打包,多机多卡则…...

Wiki.js日志系统终极指南:从记录到安全监控的全面解析

Wiki.js日志系统终极指南:从记录到安全监控的全面解析 【免费下载链接】wiki- Wiki.js | A modern and powerful wiki app built on Node.js 项目地址: https://gitcode.com/GitHub_Trending/wiki78/wiki- 当您的团队在Wiki.js中协作编辑文档时,是…...

高效批量OCR处理实战指南:提升图片文字提取效率的完整方案

高效批量OCR处理实战指南:提升图片文字提取效率的完整方案 【免费下载链接】Umi-OCR Umi-OCR: 这是一个免费、开源、可批量处理的离线OCR软件,适用于Windows系统,支持截图OCR、批量OCR、二维码识别等功能。 项目地址: https://gitcode.com/…...

普里斯特利时间轴制作教程:使用chart-doctor创建精美历史时间线

普里斯特利时间轴制作教程:使用chart-doctor创建精美历史时间线 【免费下载链接】chart-doctor Sample files to accompany the FTs Chart Doctor column 项目地址: https://gitcode.com/gh_mirrors/ch/chart-doctor chart-doctor是GitHub加速计划中的一个实…...

10个ProjectLearn性能优化技巧:提升网站加载速度和用户体验的终极指南

10个ProjectLearn性能优化技巧:提升网站加载速度和用户体验的终极指南 【免费下载链接】projectlearn-project-based-learning A curated list of project tutorials for project-based learning. 项目地址: https://gitcode.com/gh_mirrors/pr/projectlearn-proj…...