存储整数的实际内存成本是多少?
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 的算法的可行性,而且它有点占用内存。
您的问题很难回答,因为确切的内存占用量取决于几个因素,我将在下面概述其中的一些因素。
如果您只需要一些快速数字,请考虑:
memory_get_usage
— Returns分配给PHP 的内存量
memory_get_peak_usage
— Returns 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 引擎的内部更改。有关详细信息,请参阅这两篇博文:
- https://nikic.github.io/2015/05/05/Internal-value-representation-in-PHP-7-part-1.html
- https://nikic.github.io/2015/06/19/Internal-value-representation-in-PHP-7-part-2.html
另请参阅这些资源以获取更多信息:
假设我只是这样做
$arr = array();
for ($i = 0; $i < 10; $i++) $arr[] = $i;
所以我将 10 个整数存储在一个数组中。如果整数是 32b,内存成本应该是 40 字节。问题是,我没有告诉 php 它是一个整数,所以它要么必须将它存储为其他东西? (例如 js 喜欢做双打)或保留额外的数据以提醒自己它是什么?这 10 个数字 + 数组在系统内存中实际占用了多少内存?
它很重要,因为我正在尝试评估我想移植到 php 的算法的可行性,而且它有点占用内存。
您的问题很难回答,因为确切的内存占用量取决于几个因素,我将在下面概述其中的一些因素。
如果您只需要一些快速数字,请考虑:
memory_get_usage
— Returns分配给PHP 的内存量
memory_get_peak_usage
— Returns 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 引擎的内部更改。有关详细信息,请参阅这两篇博文:
- https://nikic.github.io/2015/05/05/Internal-value-representation-in-PHP-7-part-1.html
- https://nikic.github.io/2015/06/19/Internal-value-representation-in-PHP-7-part-2.html
另请参阅这些资源以获取更多信息: