ホームページ >バックエンド開発 >PHPチュートリアル >PHPスクリプトのメモリ管理とガベージコレクション

PHPスクリプトのメモリ管理とガベージコレクション

不言
不言オリジナル
2018-05-30 09:57:411714ブラウズ

この記事では、PHP スクリプトのメモリ管理とガベージ コレクションについて、例を通して説明します。必要な方は参考にしてください。

参照の割り当て

$a = 'apple';
$b = &$a;

上記のコードでは、文字列を変数 a に割り当て、次に a の参照を変数 b に割り当てます。明らかに、この時点で指すメモリは次のようになるはずです:

$a -> 'apple' <- $b

a と b は同じメモリ領域 (変数コンテナ) を指し、var_dump($a, $b)</code を通じて <code> を取得します。 > string(5) "apple" string(5) "apple" 、これは予期された結果です。 var_dump($a, $b) 得到 string(5) "apple" string(5) "apple" ,这是我们预期的结果。


引用计数 与 unset( )

假如我想将 &#39;apple&#39; 这个字符串从内存中释放掉。我是这么做的:

unset($a);

但是通过再次打印 $a $b 两变量的信息,我得到了这样的结果:Notice: Undefined variable: astring(5) "apple" 。奇怪,$a $b 同时指向一块内存区域,又明明将$a释放了,为什么$b还是&#39;apple&#39;

其实是这样的,unset()这是将一个变量指针销毁了,并没有释放掉那块内存区域(变量容器),所以执行完操作之后,内存指向只是变成了这样:

&#39;apple&#39; <- $b

要牢记重点: unset()并没有释放变量所指向的那块内存(变量容器),而只是将变量指针销毁了。同时,将那块内存的 引用计数 (ref count) 减1,当引用计数为0时,也就是说当那块内存(变量容器)不被任何变量引用时,便会触发php的垃圾回收。

用代码来验证一下:

$a = &#39;apple&#39;;
$b = &$a;

$before = memory_get_usage();
unset($a);
$after = memory_get_usage();

var_dump($before - $after);  // 结果为int(0),没有释放
$a = &#39;apple&#39;;
$b = &$a;

$before = memory_get_usage();
unset($a, $b);
$after = memory_get_usage();

var_dump($before - $after);  // 结果为int(24),得到释放

直接回收

那要怎样做才能真正释放掉 &#39;apple&#39; 所占用的内存呢?

利用上述方法,我们可以在 unset($a) 之后再 unset($b) ,将内存区域的所有引用都销毁,引用计数减为0了,自然就被php回收了。

当然,还有更直接的方法:

$a = null;

直接赋值 null 会将将 $a 所指向的内存区域置空,并将引用计数归零,内存便被释放。


脚本执行结束

php是脚本语言,当脚本执行结束之后,脚本内使用的所有内存都会被释放。那么,我们手动去释放内存有意义吗?其实关于这个问题,早有解答,推荐大家看一下鸟哥 @laruence 2012年发表的一篇文章:请手动释放你的资源(Please release resources manually)

引用赋值

$a = &#39;apple&#39;;
$b = &$a;

上述代码中,我将一个字符串赋值给变量a,然后将a的引用赋值给了变量b。显然,这个时候的内存指向应该是这样的:

$a -> 'apple' <- $b

a和b指向了同一块内存区域(变量容器),我们通过 var_dump($a, $b) 得到 string(5) "apple" string(5) "apple" ,这是我们预期的结果。


引用计数 与 unset( )

假如我想将 'apple' 这个字符串从内存中释放掉。我是这么做的:

unset($a);

但是通过再次打印 $a $b 两变量的信息,我得到了这样的结果:Notice: Undefined variable: astring(5) "apple" 。奇怪,$a $b 同时指向一块内存区域,又明明将$a释放了,为什么$b还是'apple'

其实是这样的,unset()这是将一个变量指针销毁了,并没有释放掉那块内存区域(变量容器),所以执行完操作之后,内存指向只是变成了这样:

'apple' <- $b

要牢记重点: unset()并没有释放变量所指向的那块内存(变量容器),而只是将变量指针销毁了。同时,将那块内存的 引用计数 (ref count) 减1,当引用计数为0时,也就是说当那块内存(变量容器)不被任何变量引用时,便会触发php的垃圾回收。

用代码来验证一下:

$a = 'apple';
$b = &$a;

$before = memory_get_usage();
unset($a);
$after = memory_get_usage();

var_dump($before - $after);  // 结果为int(0),没有释放
$a = 'apple';
$b = &$a;

$before = memory_get_usage();
unset($a, $b);
$after = memory_get_usage();

var_dump($before - $after);  // 结果为int(24),得到释放

直接回收

那要怎样做才能真正释放掉 'apple' 所占用的内存呢?

利用上述方法,我们可以在 unset($a) 之后再 unset($b) ,将内存区域的所有引用都销毁,引用计数减为0了,自然就被php回收了。

当然,还有更直接的方法:

$a = null;

直接赋值 null 会将将 $a


参照カウントと unset()🎜🎜 文字列 'apple' をメモリから解放したいとします。これが私がやったことです: 🎜rrreee🎜 しかし、2つの変数 $a $b の情報を再度出力すると、次の結果が得られました: Notice: 未定義の変数: astring(5) "apple"。奇妙なことに、$a $b は同時にメモリ領域を指しており、明らかに $a を解放しています。なぜ $b は のままなのでしょうか。 >「リンゴ」。 🎜🎜実際には、これが当てはまります。 unset() は、変数ポインタを破棄し、メモリ領域 (変数コンテナ) を解放しません。 したがって、操作が完了すると、メモリ ポインタは次のようになります。 : 🎜rrreee🎜覚えておくべき重要なポイント: unset() は変数が指すメモリ (変数コンテナ) を解放せず、変数ポインタを破棄するだけです。同時に、 そのメモリの参照カウント (ref count) が 1 減らされます。参照カウントが 0 のとき、つまり、そのメモリ (変数コンテナ) のとき、変数によって参照されていない場合、PHP のガベージ コレクションがトリガーされます。 🎜🎜コードで確認してください: 🎜rrreeerrreee
🎜直接リサイクル🎜🎜それでは、'apple' によって占有されているメモリを実際に解放するにはどうすればよいでしょうか? 🎜🎜上記のメソッドを使用すると、unset($a) してから unset($b) を実行して、メモリ領域内のすべての参照を破棄し、参照カウントを 0 に減らすことができます。 、当然phpによってリサイクルされます。 🎜🎜もちろん、もっと直接的な方法もあります: 🎜rrreee🎜 null を直接代入すると、$a が指すメモリ領域が空になり、参照カウントがゼロにリセットされます。 、メモリが解放されます。 🎜
🎜スクリプトの実行が終了します🎜🎜phpは、スクリプトの実行が終了すると、スクリプトで使用されていたすべてのメモリが解放されます。では、手動でメモリを解放するのは意味があるのでしょうか?実際、この質問は長い間解決されてきました。2012 年に Brother Bird @laruence によって公開された記事を読むことをお勧めします。リソースを手動で解放してください (リソースを手動で解放してください)🎜🎜参照の割り当て🎜rrreee🎜 , 変数aに文字列を代入し、変数bにaの参照を代入します。明らかに、この時点で指すメモリは次のようになるはずです: 🎜rrreee🎜a と b は同じメモリ領域 (変数コンテナ) を指し、var_dump($a, $b)を通じて を取得します。 > string(5) "apple" string(5) "apple" 、これは予期された結果です。 🎜
🎜参照カウントと unset()🎜🎜 文字列 'apple' をメモリから解放したいとします。これが私がやったことです: 🎜rrreee🎜 しかし、2つの変数 $a $b の情報を再度出力すると、次の結果が得られました: Notice: 未定義の変数: astring(5) "apple"。奇妙なことに、$a $b は同時にメモリ領域を指しており、明らかに $a を解放しています。なぜ $b は のままなのでしょうか。 >「リンゴ」。 🎜🎜実際には、これが当てはまります。 unset() は、変数ポインタを破棄し、メモリ領域 (変数コンテナ) を解放しません。 したがって、操作が完了すると、メモリ ポインタは次のようになります。 : 🎜rrreee🎜覚えておくべき重要なポイント: unset() は変数が指すメモリ (変数コンテナ) を解放せず、変数ポインタを破棄するだけです。同時に、 そのメモリの参照カウント (ref count) が 1 減らされます。参照カウントが 0 のとき、つまり、そのメモリ (変数コンテナ) のとき、変数によって参照されていない場合、PHP のガベージ コレクションがトリガーされます。 🎜🎜コードで確認してください: 🎜rrreeerrreee
🎜直接リサイクル🎜🎜それでは、'apple' によって占有されているメモリを実際に解放するにはどうすればよいでしょうか? 🎜🎜上記のメソッドを使用すると、unset($a) してから unset($b) を実行して、メモリ領域内のすべての参照を破棄し、参照カウントを 0 に減らすことができます。 、当然phpによってリサイクルされます。 🎜🎜もちろん、もっと直接的な方法もあります: 🎜rrreee🎜 null を直接代入すると、$a が指すメモリ領域が空になり、参照カウントがゼロにリセットされます。 、メモリが解放されます。 🎜

以上がPHPスクリプトのメモリ管理とガベージコレクションの詳細内容です。詳細については、PHP 中国語 Web サイトの他の関連記事を参照してください。

声明:
この記事の内容はネチズンが自主的に寄稿したものであり、著作権は原著者に帰属します。このサイトは、それに相当する法的責任を負いません。盗作または侵害の疑いのあるコンテンツを見つけた場合は、admin@php.cn までご連絡ください。