DateFormatter 中的内存泄漏
Posted
技术标签:
【中文标题】DateFormatter 中的内存泄漏【英文标题】:Memory leak in DateFormatter 【发布时间】:2019-07-26 20:47:32 【问题描述】:我经常在代码中使用DateFormatter
并没有发现任何问题,但后来我决定测试一个类进行大量迭代 - 我注意到一个巨大的内存泄漏。
泄露的原因是DateFormatter
。
简单代码:
var dateFormatter:DateFormatter
dateFormatter = DateFormatter()
let date : Date = Date()
for _ in 1...100000000
_ = dateFormatter.string(from: date)
没有泄漏。 5.9 MB 每个进程所有运行时间。
但是当我使用dateFormat
:
var dateFormatter:DateFormatter
dateFormatter = DateFormatter()
let date : Date = Date()
dateFormatter!.dateFormat = "yyyyMMddHHmmssSSS"
for _ in 1...100000000
_ = dateFormatter.string(from: date)
在运行结束时使用内存 4 GB。
如果我使用:
dateFormatter.dateFormat = "yyyyMMdd"
没有泄漏。
如果我使用:
dateFormatter.dateFormat = "yyyyMMddHH"
泄漏。
【问题讨论】:
第二个问题是执行时间。我不得不使用NSDate()。 TimeIntervalSince1970 用完的内存和泄漏是根本不同的事情。 你是在操场上测试这个吗? 配置文件标记这是内存泄漏)像控制台应用程序一样测试) 【参考方案1】:在您的第一个示例中,您正在创建零长度字符串。修改您的代码以打印出来;您在日志中看不到任何内容。尝试将日期格式化程序的输出分配给变量并通过在循环中放置断点来检查其值;该值始终为空字符串。这可以解释为什么内存使用率仍然如此之低。
在您的第二个示例中,您正在按照日期格式创建字符串。和以前一样,检查日期格式化程序中的值,你会看到。这解释了为什么您会迅速耗尽物理 RAM:您在该循环中创建了 1 亿个非空字符串。
即使在第一个示例中,这也不是最佳做法。很有可能,如果你真的需要 1 亿个字符串,你可以对它们做一些事情,比如将它们放在一个在其他地方实例化的数组中,或者诸如此类。你不想让它们堆积在那个巨大的循环中。
这是您的代码的修改版本:
let date = Date()
let dateFormatter = DateFormatter()
dateFormatter.dateFormat = "yyyyMMddHHmmssSSS"
for _ in 1...100000000
autoreleasepool(invoking: () -> () in
let string = dateFormatter.string(from: date)
print("\(string)")
)
请注意,我在循环中使用了 autoreleasepool 闭包。这确保了我在每次循环中创建的字符串在下一次迭代之前被释放。在那个闭包中,你可以用那个字符串做任何你想做的事情。您可以在其中创建任意数量的值,并且它们都将在循环的下一次迭代之前被释放。在我的机器上,所有东西都在消耗 16.2 MB 的内存。您的里程可能会有所不同。
祝您一切顺利。
【讨论】:
好的)第一个例子 - 我的错))但是当使用 dateFormatter.dateFormat = "yyyyMMdd" 没有自动释放池时 - 使用 "yyyyMMddHH" 时没有泄漏 - 大量内存使用或泄漏(配置文件标记这是泄漏) 您要我评论 DateFormatter 的内部结构。我无法解释它的行为。但无论如何,如果你在循环中实例化对象,你应该在 autoreleasepool 闭包中这样做,以确保它们在下一次迭代之前被释放。以上是关于DateFormatter 中的内存泄漏的主要内容,如果未能解决你的问题,请参考以下文章