목차
zval의 구조
순환 참조로 인한 메모리 누수
객체와 배열의 재활용 프로세스
가비지 수집의 원리
Example
백엔드 개발 PHP7 php7에서 가비지 수집 메커니즘을 구현하는 방법

php7에서 가비지 수집 메커니즘을 구현하는 방법

May 24, 2021 am 09:21 AM
php7 쓰레기 수집 메커니즘

이 글에서는 php7에서 가비지 수집 메커니즘을 구현하는 방법을 소개합니다. 도움이 필요한 친구들이 모두 참고할 수 있기를 바랍니다.

php7에서 가비지 수집 메커니즘을 구현하는 방법

PHP GC를 이해할 때 PHP 변수의 기본 구현을 소개할 필요가 있다고 생각합니다.

zval의 구조

// php 变量对于的c结构体
struct _zval_struct {
    zend_value value;
    union {
       ……
    } u1;
    union {
        ……
    } u2;
};
로그인 후 복사

주로 가비지 컬렉션에 대해 이야기하기 때문에 u1 u2 공용체의 기능에 대해 간략하게 소개합니다
u1 구조가 비교적 복잡해서 주로 변수 유형을 식별하는 데 사용되는 것 같아요
u2 대부분은 보조 필드, 변수의 내부 기능 구현, 캐시 친화성 향상 등입니다.
다음은 우리의 주인공

zend_value 구조체에 내장된 공용체이기도 합니다

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;
로그인 후 복사

참조 횟수 zend_refcounted *counted 는 다음에 기록됩니다. zval 값 이 유형도 가비지 수집 메커니즘을 기반으로 합니다.

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;
로그인 후 복사

모든 복잡한 유형 정의는 zend_refcounted_h 구조로 시작합니다. 이 구조에는 GC 관련 구조도 있으므로 GC 재활용을 수행할 때 GC는 모든 유형을 신경 쓸 필요가 없습니다. zend_refcounted* 구조로 처리할 수 있습니다.
#변수의 자동 재활용

PHP의 배열 및 객체 유형 변수를 제외하면 대부분의 변수는 자동으로 재활용됩니다.
PHP 일반 변수의 재활용은 변수에 대한 참조 수와 관련이 있습니다. .

공식 예시

$a = 1;
$b = $a;
xdebug_debug_zval('a');
$a =10;
xdebug_debug_zval('a');
unset($a);
xdebug_debug_zval('a');
로그인 후 복사

결과

a:
(refcount=2, is_ref=0),int 1
a:
(refcount=1, is_ref=0),int 10
a: no such symbol
로그인 후 복사

$a =10이 PHP의 COW(기록 중 복사) 메커니즘과 관련될 때 $b는 원본 $a를 복사하여 이를 제거하는 것을 볼 수 있습니다. a이므로 a의 참조 수(refcount)는 1로 줄어듭니다.

그러면 ($a)를 사용한 후에 a에 대한 참조 횟수는 0이 됩니다. 이는 가비지 변수로 간주되어 공간을 확보합니다.

또 다른 예

$a = [1];
$a[1] = &$a;
unset($a);
로그인 후 복사

unset($a) 전의 $a 타입은 참조 타입

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) 이후에는 이렇게 됩니다

이때, we unset 작업 중에 refcount가 2에서 1로 변경됩니다. $a를 가리키는 내부 참조가 있기 때문에 해당 참조가 차지하는 공간은 외부적으로 파괴되지 않습니다.

그러면 외부 참조가 손상되어 사용할 수 없게 되었습니다. 이는 C 언어에서 와일드 포인터(Wild Pointer)라고 불리는 "고아(orphan)"가 됩니다. PHP에서는 이를 순환 참조라고 합니다. 메모리 누수. 변수를 삭제하려면 PHP 스크립트가 끝날 때까지 기다리면 됩니다.

순환 참조로 인한 메모리 누수

이러한 쓰레기를 정리하기 위해 두 가지 지침이 도입되었습니다

  • 참조 횟수가 0으로 줄어들면 변수 컨테이너가 가비지가 아닌 클리어(무료)됩니다

  • zval의 참조 횟수가 감소한 후에도 여전히 0보다 큰 경우 가비지 사이클에 들어갑니다. 둘째, 가비지 사이클 동안 참조 횟수가 1만큼 감소하는지 확인하고 참조가 0인 변수 컨테이너를 확인하여 어떤 부분이 가비지인지 알아냅니다.

순환 참조는 기본적으로 배열과 객체에만 나타납니다. 객체 자체가 참조이기 때문입니다.

객체와 배열의 재활용 프로세스

PHP7의 가비지 컬렉션은 두 부분으로 구성됩니다. 하나는 가비지 컬렉터이고, 하나는 가비지 수집 알고리즘입니다.

가비지 수집기는 방금 언급한 쓰레기일 수 있는 요소를 재활용 풀로 수집합니다. 즉, 변수 zend_refcount>0이 재활용 풀에 배치됩니다. 재활용 풀의 가치가 일정 금액에 도달하면 균일하게 통과됩니다. 시뮬레이션된 삭제를 수행합니다. zend_refcount=0인 경우 가비지로 간주되어 직접 삭제됩니다.

재활용 풀의 모든 변수를 순회한 다음 각 변수를 기반으로 각 멤버를 순회합니다. 멤버가 여전히 중첩되어 있으면 계속해서 순회하세요. 그런 다음 모든 구성원의 시뮬레이션된 참조 횟수를 -1로 설정합니다. 이때 외부변수의 참조카운트가 0인 경우. 그렇다면 그것은 분명히 쓰레기로 간주될 수 있다. 0보다 크면 참조 수가 복원되고 가비지 수집 풀에서 제거됩니다.

가비지 수집의 원리

변수가 가비지가 아닌 경우 모든 멤버 변수의 참조가 1만큼 줄어들면 전체 변수 참조는 확실히 0이 되지 않습니다.

Example

설명하기가 좀 어려우니 예를 들어보겠습니다. 처음 sf.gg를 검색했을 때 GC에 대한 질문을 보고 답변을 드렸습니다. GC 가비지 수집 메커니즘에 관해

주제는 다음과 같습니다
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 
模拟减结束,那么此变量被当成垃圾回收。
로그인 후 복사

추천 학습:php 비디오 튜토리얼

위 내용은 php7에서 가비지 수집 메커니즘을 구현하는 방법의 상세 내용입니다. 자세한 내용은 PHP 중국어 웹사이트의 기타 관련 기사를 참조하세요!

본 웹사이트의 성명
본 글의 내용은 네티즌들의 자발적인 기여로 작성되었으며, 저작권은 원저작자에게 있습니다. 본 사이트는 이에 상응하는 법적 책임을 지지 않습니다. 표절이나 침해가 의심되는 콘텐츠를 발견한 경우 admin@php.cn으로 문의하세요.

핫 AI 도구

Undresser.AI Undress

Undresser.AI Undress

사실적인 누드 사진을 만들기 위한 AI 기반 앱

AI Clothes Remover

AI Clothes Remover

사진에서 옷을 제거하는 온라인 AI 도구입니다.

Undress AI Tool

Undress AI Tool

무료로 이미지를 벗다

Clothoff.io

Clothoff.io

AI 옷 제거제

Video Face Swap

Video Face Swap

완전히 무료인 AI 얼굴 교환 도구를 사용하여 모든 비디오의 얼굴을 쉽게 바꾸세요!

뜨거운 도구

메모장++7.3.1

메모장++7.3.1

사용하기 쉬운 무료 코드 편집기

SublimeText3 중국어 버전

SublimeText3 중국어 버전

중국어 버전, 사용하기 매우 쉽습니다.

스튜디오 13.0.1 보내기

스튜디오 13.0.1 보내기

강력한 PHP 통합 개발 환경

드림위버 CS6

드림위버 CS6

시각적 웹 개발 도구

SublimeText3 Mac 버전

SublimeText3 Mac 버전

신 수준의 코드 편집 소프트웨어(SublimeText3)

php7.0에 플러그인이 설치되어 있는데 여전히 설치되지 않은 것으로 표시되면 어떻게 해야 합니까? php7.0에 플러그인이 설치되어 있는데 여전히 설치되지 않은 것으로 표시되면 어떻게 해야 합니까? Apr 02, 2024 pm 07:39 PM

PHP 7.0에서 설치된 플러그인이 표시되지 않는 문제를 해결하려면: 플러그인 구성을 확인하고 플러그인을 활성화하세요. 구성 변경 사항을 적용하려면 PHP를 다시 시작하세요. 플러그인 파일 권한이 올바른지 확인하세요. 플러그인이 올바르게 작동하도록 하려면 누락된 종속성을 설치하세요. 다른 모든 단계가 실패하면 PHP를 다시 빌드하세요. 다른 가능한 원인으로는 호환되지 않는 플러그인 버전, 잘못된 버전 로드 또는 PHP 구성 문제 등이 있습니다.

php7.0에 mongo 확장을 설치하는 방법 php7.0에 mongo 확장을 설치하는 방법 Nov 21, 2022 am 10:25 AM

php7.0에서 mongo 확장을 설치하는 방법: 1. mongodb 사용자 그룹 및 사용자를 생성합니다. 2. mongodb 소스 코드 패키지를 다운로드하고 소스 코드 패키지를 "/usr/local/src/" 디렉터리에 넣습니다. "src/" 디렉터리를 입력합니다. 4. 소스 코드 패키지의 압축을 풉니다. 5. mongodb 파일 디렉터리를 생성합니다. 6. "mongodb/" 디렉터리에 파일을 복사합니다. 7. mongodb 구성 파일을 생성하고 구성을 수정합니다.

php7에서 tcp 포트가 작동하지 않는 것을 감지했을 때 문제를 해결하는 방법 php7에서 tcp 포트가 작동하지 않는 것을 감지했을 때 문제를 해결하는 방법 Mar 22, 2023 am 09:30 AM

php5에서는 fsockopen() 함수를 사용하여 TCP 포트를 감지할 수 있습니다. 이 기능은 네트워크 연결을 열고 일부 네트워크 통신을 수행하는 데 사용할 수 있습니다. 그러나 php7에서는 fsockopen() 함수에 포트를 열 수 없거나 서버에 연결할 수 없는 등의 몇 가지 문제가 발생할 수 있습니다. 이 문제를 해결하기 위해, 우리는 소켓_create() 함수와 소켓_연결() 함수를 사용하여 TCP 포트를 감지할 수 있습니다.

PHP 서버 환경 FAQ 가이드: 일반적인 문제를 신속하게 해결 PHP 서버 환경 FAQ 가이드: 일반적인 문제를 신속하게 해결 Apr 09, 2024 pm 01:33 PM

PHP 서버 환경에 대한 일반적인 솔루션에는 올바른 PHP 버전이 설치되어 있는지, 관련 파일이 모듈 디렉터리에 복사되었는지 확인하는 것이 포함됩니다. SELinux를 일시적으로 또는 영구적으로 비활성화합니다. 필요한 확장이 추가되고 올바르게 설정되었는지 확인하려면 PHP.ini를 확인하고 구성하십시오. PHP-FPM 서비스를 시작하거나 다시 시작합니다. 해결 문제에 대해서는 DNS 설정을 확인하세요.

php7.0을 설치하고 배포하는 방법 php7.0을 설치하고 배포하는 방법 Nov 30, 2022 am 09:56 AM

php7.0 설치 및 배포 방법: 1. PHP 공식 웹사이트로 이동하여 로컬 시스템에 해당하는 설치 버전을 다운로드합니다. 2. 다운로드한 zip 파일을 지정된 디렉터리에 추출합니다. 3. 명령줄 창을 열고 다음으로 이동합니다. "E:\php7" 디렉토리 "php -v" 명령을 실행하세요.

PHP의 가비지 수집 메커니즘에 대해 설명하는 기사 PHP의 가비지 수집 메커니즘에 대해 설명하는 기사 Aug 26, 2022 am 10:48 AM

이 기사는 PHP의 가비지 수집 메커니즘에 대한 심층적인 이해를 제공할 것입니다. 도움이 되기를 바랍니다.

시스템 재시작 후 UnixSocket의 권한을 자동으로 설정하는 방법은 무엇입니까? 시스템 재시작 후 UnixSocket의 권한을 자동으로 설정하는 방법은 무엇입니까? Mar 31, 2025 pm 11:54 PM

시스템이 다시 시작된 후 UnixSocket의 권한을 자동으로 설정하는 방법. 시스템이 다시 시작될 때마다 UnixSocket의 권한을 수정하려면 다음 명령을 실행해야합니다.

Docker 환경에서 PECL을 사용하여 확장자를 설치할 때 오류가 발생하는 이유는 무엇입니까? 그것을 해결하는 방법? Docker 환경에서 PECL을 사용하여 확장자를 설치할 때 오류가 발생하는 이유는 무엇입니까? 그것을 해결하는 방법? Apr 01, 2025 pm 03:06 PM

Docker 환경을 사용할 때 Docker 환경에 Extensions를 설치하기 위해 PECL을 사용하여 오류의 원인 및 솔루션. 종종 일부 두통이 발생합니다 ...

See all articles