获取堆上对象的大小(以字节为单位)

Ale*_*ica 10 heap-memory memory-layout dynamic-allocation swift

我知道你可以MemoryLayout<T>.size用来获得一个类型的大小T.

例如: MemoryLayout<Int32>.size // 4

但是,对于类实例(对象),MemoryLayout<T>.size返回对象的引用大小(64位计算机上的8个字节),而不是堆上实际对象的大小.

class ClassA { // Objects should be at least 8 bytes
    let x: Int64 = 0
}

class ClassB  {// Objects should be at least 16 bytes
    let x: Int64 = 0
    let y: Int64 = 0
}

MemoryLayout<ClassA>.size // 8
MemoryLayout<ClassB>.size // 8, as well :(
Run Code Online (Sandbox Code Playgroud)

如何获得物体的大小?

对于那些想知道的人,我没有真正的需要,我只是在探索Swift及其与C的互操作性.

OOP*_*Per 15

据我在Playground测试,这个函数返回看似很重要的值:

func heapSize(_ obj: AnyObject) -> Int {
    return malloc_size(Unmanaged.passRetained(obj).toOpaque())
}

class MyClass {
    //no properites...
}
let myObj = MyClass()
print(heapSize(myObj)) //->16

class MyBiggerClass {
    var str: String?
    var i: Int = 0
}
let myBiggerObj = MyBiggerClass()
print(heapSize(myBiggerObj)) //->64
Run Code Online (Sandbox Code Playgroud)

似乎当前的Swift运行时使用了malloc兼容的东西来在堆中分配内存.(malloc在分配小块时,给出一些填充以适合分配的大小以供给2.因此,实例所需的实际大小可能小于malloc_size.)

我还没有测试过它可以工作多远,而且根据当前实现的未记录行为在未来的任何时候都会发生变化,而不会发出任何通知.

但如果你真的知道这一点,这可能是一个很好的探索起点.


Ham*_*ish 11

Apple平台上的一个选项,因为Swift类目前是在Objective-C类之上构建的,它将使用Obj-C运行时函数class_getInstanceSize,该函数为您提供类实例的字节大小,包括任何填充.

// on a 64-bit machine (1 word == 8 bytes)...

import Foundation

class C {}
print(class_getInstanceSize(C.self)) // 16 bytes metadata for empty class 
                                     // (isa ptr + ref count)

class C1 {
    var i = 0
    var i1 = 0
    var b = false
}

print(class_getInstanceSize(C1.self)) // 40 bytes
// (16 metadata + 24 ivars, 8 for i + 8 for i1 + 1 for b + 7 padding)
Run Code Online (Sandbox Code Playgroud)