Maison  >  Article  >  développement back-end  >  Comment implémenter le mécanisme de garbage collection en php7

Comment implémenter le mécanisme de garbage collection en php7

醉折花枝作酒筹
醉折花枝作酒筹avant
2021-05-24 09:21:221800parcourir

Cet article vous présentera comment php7 implémente le mécanisme de garbage collection. Il a une certaine valeur de référence. Les amis dans le besoin peuvent s'y référer. J'espère qu'il sera utile à tout le monde.

Comment implémenter le mécanisme de garbage collection en php7

Lors de la compréhension de notre GC php, je pense qu'il est nécessaire d'introduire l'implémentation sous-jacente de nos variables php.

La structure de zval

// php 变量对于的c结构体
struct _zval_struct {
    zend_value value;
    union {
       ……
    } u1;
    union {
        ……
    } u2;
};

Comme il parle principalement de garbage collection, voici une brève introduction à la fonction de l'union u1 u2
La structure de u1 est plus compliquée. Je pense qu'il est principalement utilisé pour identifier les variables. Type
u2 La plupart d'entre eux sont des champs auxiliaires, l'implémentation de fonctions internes des variables, l'amélioration de la convivialité du cache, etc.
Vient ensuite notre protagoniste

zend_value qui est également intégré dans la structure. L'union

typedef union _zend_value {
    zend_long         lval;//整形
    double            dval;//浮点型
    zend_refcounted  *counted;//获取不同类型的gc头部
    zend_string      *str;//string字符串
    zend_array       *arr;//数组
    zend_object      *obj;//对象
    zend_resource    *res;//资源
    zend_reference   *ref;//是否是引用类型
  
    // 忽略下面的结构,与我们讨论无关
    zend_ast_ref     *ast;
    zval             *zv;
    void             *ptr;
    zend_class_entry *ce;
    zend_function    *func;
    struct {
        ZEND_ENDIAN_LOHI(
            uint32_t w1,
            uint32_t w2)
    } ww;
} zend_value;

enregistre le type de nombre de références zend_refcounted *counted dans la valeur de zval. Notre mécanisme de récupération de place est également basé sur cela.

typedef struct _zend_refcounted_h {
    uint32_t         refcount;          /* reference counter 32-bit */
    union {
        struct {
            ZEND_ENDIAN_LOHI_3(
                zend_uchar    type,
                zend_uchar    flags,    /* used for strings & objects */
                uint16_t      gc_info)  /* keeps GC root number (or 0) and color */
        } v;
        uint32_t type_info;
    } u;
} zend_refcounted_h;

Toutes les définitions de types complexes commencent par la structure zend_refcounted_h En plus du comptage de références, cette structure a également des structures liées au GC. Par conséquent, lors du recyclage du GC, le GC n'a pas besoin de se soucier des détails. Quel est le type ? Toutes peuvent être traitées comme des structures zend_refcounted*.
#Recyclage automatique des variables

En PHP, à l'exception des variables de type tableau et objet, la plupart des autres sont automatiquement recyclées
Le recyclage des variables ordinaires PHP est lié au nombre de références à la variable.

Exemple officiel

$a = 1;
$b = $a;
xdebug_debug_zval('a');
$a =10;
xdebug_debug_zval('a');
unset($a);
xdebug_debug_zval('a');

Résultat

a:
(refcount=2, is_ref=0),int 1
a:
(refcount=1, is_ref=0),int 10
a: no such symbol

Vous pouvez voir que lorsque $a =10, le mécanisme COW (copie sur écriture) de php est impliqué, $b Une copie du $a original sera copiée et la relation de référence entre eux sera libérée, de sorte que le nombre de références (refcount) de a est réduit à 1.

Ensuite, le nombre de références à a devient 0 après avoir utilisét($a). Ceci sera considéré comme une variable inutile et libérera de l'espace.

Donnez un autre exemple

$a = [1];
$a[1] = &$a;
unset($a);

Le type de $a avant unset($a) est un type de référence

a:
(refcount=2, is_ref=1),
array (size=2)
  0 => (refcount=1, is_ref=0),int 1
  1 => (refcount=2, is_ref=1),
    &array<

unset($ a) Après ça, ça devient comme ça

A ce moment, quand on désactive l'opération, le refcount passe de 2 à 1, car il y a une référence interne pointant vers $a, donc là où il se trouve à l'extérieur L'espace occupé ne sera pas détruit.

Ensuite, notre référence externe a été cassée et nous ne pouvons pas l'utiliser. Il devient un "orphelin", appelé pointeur sauvage en langage C. En php, cela s'appelle une référence circulaire. Fuite de mémoire. Si vous souhaitez détruire la variable, vous ne pouvez qu'attendre la fin du script php.

Fuites de mémoire causées par des références circulaires

Afin de nettoyer ces déchets, deux directives sont introduites

  • Si le nombre de références est réduit à zéro , le conteneur de variables sera effacé (gratuit), pas poubelle

  • Si le nombre de références d'un zval est toujours supérieur à 0 après avoir été réduit, alors il entrera dans le cycle des ordures. Deuxièmement, pendant un cycle de déchets, découvrez quelles parties sont des déchets en vérifiant si le nombre de références est réduit de 1 et en vérifiant quels conteneurs de variables ont zéro référence.

Les références circulaires n'apparaissent essentiellement que dans les tableaux et les objets. L'objet est parce qu'il est lui-même une référence

Le processus de recyclage de l'objet et du tableau

Le garbage collection de PHP7 se compose de deux parties, l'une est le garbage collector et l'autre est l'algorithme de garbage collection.

Le garbage collector collecte les éléments que nous venons de mentionner qui peuvent être des déchets dans le pool de recyclage, c'est-à-dire que la variable zend_refcount>0 est placée dans le pool de recyclage. Lorsque la valeur du pool de recyclage atteint un certain montant, il sera parcouru uniformément. Effectuez une suppression simulée. Si zend_refcount=0, cela est considéré comme un déchet et supprimé directement.

Parcourez chaque variable du pool de recyclage, puis parcourez chaque membre en fonction de chaque variable. Si les membres sont toujours imbriqués, continuez le parcours. Définissez ensuite le refcount simulé de tous les membres sur -1. Si le nombre de références de la variable externe est 0 à ce moment. Cela peut alors être clairement considéré comme une poubelle. S'il est supérieur à 0, alors le nombre de références est restauré et retiré du pool de récupération de place.

Principe du garbage collection

Si votre variable n'est pas une poubelle, alors une fois les références de toutes ses variables membres réduites de un, la référence de la variable totale ne sera certainement pas 0.

Exemple

C'est difficile à dire, alors donnons un exemple. Lorsque j'ai parcouru sf.gg pour la première fois, j'ai vu une question sur GC et j'y ai répondu. À propos du mécanisme de collecte des ordures GC

Le sujet est le suivant
Comment implémenter le mécanisme de garbage collection en php7

//我的回答
1、只要zval.value的refcount减一,然后缺其refcount的值不为0那么它就可能是垃圾,进入垃圾周期。
2、进入垃圾池遍历所有成员,包括其嵌套的成员,都对其做 refcount-1的操作,看外部的引用是否为0。

那么对于 题主的问题来说,
首先,你要想$a为垃圾,一定要先对 unset($a)操作,那么此时 $a的 refcount = 2
对于$a[0] refcount-1 不影响外部的$a,
$a[1] refcount-1 ,此时 $a的 refount=1
$a[2] refcount-1 ,此时 $a 的 refount=0 
模拟减结束,那么此变量被当成垃圾回收。

Apprentissage recommandé :Tutoriel vidéo php

Ce qui précède est le contenu détaillé de. pour plus d'informations, suivez d'autres articles connexes sur le site Web de PHP en chinois!

Déclaration:
Cet article est reproduit dans:. en cas de violation, veuillez contacter admin@php.cn Supprimer