存储整数的实际内存成本是多少?

What is the actual memory cost of storing an integer?

假设我只是这样做

$arr = array();
for ($i = 0; $i < 10; $i++) $arr[] = $i;

所以我将 10 个整数存储在一个数组中。如果整数是 32b,内存成本应该是 40 字节。问题是,我没有告诉 php 它是一个整数,所以它要么必须将它存储为其他东西? (例如 js 喜欢做双打)或保留额外的数据以提醒自己它是什么?这 10 个数字 + 数组在系统内存中实际占用了多少内存?

它很重要,因为我正在尝试评估我想移植到 php 的算法的可行性,而且它有点占用内存。

您的问题很难回答,因为确切的内存占用量取决于几个因素,我将在下面概述其中的一些因素。

如果您只需要一些快速数字,请考虑:

PHP 在内部存储值 structure called the zval:

121 struct _zval_struct {
122    zend_value        value;            /* value */
123    union {
124        struct {
125            ZEND_ENDIAN_LOHI_4(
126                zend_uchar    type,         /* active type */
127                zend_uchar    type_flags,
128                zend_uchar    const_flags,
129                zend_uchar    reserved)     /* call info for EX(This) */
130        } v;
131        uint32_t type_info;
132    } u1;
133    union {
134        uint32_t     var_flags;
135        uint32_t     next;                 /* hash collision chain */
136        uint32_t     cache_slot;           /* literal cache slot */
137        uint32_t     lineno;               /* line number (for ast nodes) */
138        uint32_t     num_args;             /* arguments number for EX(This) */
139        uint32_t     fe_pos;               /* foreach position */
140        uint32_t     fe_iter_idx;          /* foreach iterator index */
141    } u2;
142};

这是一个联合类型,意味着它可以存储多种类型的值。所以是的,它会保留额外的数据来提醒自己它是什么。整数通常表示为长整数(32 位或 64 位,具体取决于您的平台)。

关于数组,有一个excellent blog post by NikiC给出了详细的解释。要点:

                              |  64 bit   | 32 bit
---------------------------------------------------
zval                         |  24 bytes | 16 bytes
+ cyclic GC info             |   8 bytes |  4 bytes
+ allocation header          |  16 bytes |  8 bytes
===================================================
zval (value) total           |  48 bytes | 28 bytes
===================================================
bucket                       |  72 bytes | 36 bytes
+ allocation header          |  16 bytes |  8 bytes
+ pointer                    |   8 bytes |  4 bytes
===================================================
bucket (array element) total |  96 bytes | 48 bytes
===================================================
total total                  | 144 bytes | 76 bytes

The above numbers will vary depending on your operating system, your compiler and your compile options. E.g. if you compile PHP with debug or with thread-safety, you will get different numbers. But I think that the sizes given above are what you will see on an average 64-bit production build of PHP 5.3 on Linux.

还值得注意的是,非数字数组键标签也是一种分配,因此标签越长,它们消耗的内存就越多。

以上数字还取决于您使用的 PHP 版本。 PHP >= 7 的内存占用应该比任何 PHP < 7 的内存占用少得多,因为 Zend 引擎的内部更改。有关详细信息,请参阅这两篇博文:

另请参阅这些资源以获取更多信息: