Przechwytywanie (zewnętrznie) zużycia pamięci danego wywołania zwrotnego

Problem

Powiedzmy, że mam tę funkcję:

function hog($i = 1) // uses $i * 0.5 MiB, returns $i * 0.25 MiB
{
    $s = str_repeat('a', $i * 1024 * 512); return substr($s, $i * 1024 * 256);
}

Chciałbym to nazwać i móc sprawdzić maksymalną ilość używanej pamięci.

Innymi słowy:memory_get_function_peak_usage($callback);. czy to możliwe?

Co próbowałem

Używam następujących wartości jako moich nie Monotonicznie rosnących$i argument zahog():

$iterations = array_merge(range(0, 50, 10), range(50, 0, 5));
$iterations = array_fill_keys($iterations, 0);

Co jest zasadniczo:

(
    [0] => 0
    [10] => 0
    [20] => 0
    [30] => 0
    [40] => 0
    [50] => 0
    [45] => 0
    [35] => 0
    [25] => 0
    [15] => 0
    [5] => 0
)
Zamknięciememory_get_usage()
foreach ($iterations as $key => $value)
{
    $alpha = memory_get_usage(); hog($key);
    $iterations[$key] = memory_get_usage() - $alpha;
}

print_r($iterations);

Wydajność:

(
    [0] => 96
    [10] => 0
    [20] => 0
    [30] => 0
    [40] => 0
    [50] => 0
    [45] => 0
    [35] => 0
    [25] => 0
    [15] => 0
    [5] => 0
)

Jeśli zachowam wartość zwracanąhog(), wyniki zaczynają wyglądać bardziej realistycznie:

foreach ($iterations as $key => $value)
{
    $alpha = memory_get_usage(); $s = hog($key);
    $iterations[$key] = memory_get_usage() - $alpha; unset($s);
}

print_r($iterations);

Wydajność:

(
    [0] => 176
    [10] => 2621536
    [20] => 5242976
    [30] => 7864416
    [40] => 10485856
    [50] => 13107296
    [45] => 11796576
    [35] => 9175136
    [25] => 6553696
    [15] => 3932256
    [5] => 1310816
)

Zgodnie z oczekiwaniami, teraz pokazuje mi ilość pamięcizwrócony, ale potrzebuję całej używanej pamięci.

Za pomocąregister_tick_function():

Nie wiedziałem, ale okazuje się, że kiedy to robisz:

declare (ticks=1)
{
    $a = hog(1);
}

Nie będzie zaznaczać każdej linii, oświadczenia ani bloku kodu wewnątrzhog() funkcja, tylko dla kodu wewnątrzdeclare blok - więc jeśli funkcja nie jest w niej zdefiniowana, ta opcja jest bezużyteczna.

Mieszanie zgc_* Funkcje:

Próbowałem (bez większej nadziei muszę powiedzieć) używając kombinacjigc_disable(), gc_enable() igc_collect_cycles() z obydwoma eksperymentami powyżej, aby zobaczyć, czy coś się zmieniło - nie.

questionAnswers(3)

yourAnswerToTheQuestion