백엔드 개발 PHP 튜토리얼 PHP 배열의 낮은 메모리 활용도에 대한 자세한 분석

PHP 배열의 낮은 메모리 활용도에 대한 자세한 분석

Jan 30, 2018 am 10:11 AM
php 메모리

PHP는 실제로 그렇게 많은 메모리를 소비합니까? 그래서 이번 기회에 PHP의 데이터형 구현에 대해 알아보았습니다. 이 기사에서는 주로 낮은 메모리 사용률과 취약한 유형의 PHP 배열에 대한 자세한 해석을 소개합니다. 이는 특정 참조 가치가 있으며 관심 있는 친구들이 참조할 수 있습니다.

먼저 테스트해 보겠습니다.


<?php 
  echo memory_get_usage() , &#39;<br>&#39;; 
  $start = memory_get_usage(); 
  $a = Array(); 
  for ($i=0; $i<1000; $i++) { 
   $a[$i] = $i + $i; 
  } 
  $end = memory_get_usage(); 
  echo memory_get_usage() , &#39;<br>&#39;; 
  echo &#39;argv:&#39;, ($end - $start)/1000 ,&#39;bytes&#39; , &#39;<br>&#39;;
로그인 후 복사

결과:

353352
437848
argv: 84.416바이트

1000개 요소의 정수 배열은 메모리를 소비합니다(4378 48 - 353352) 바이트, 약 82KB, 이는 각 요소는 84바이트의 메모리를 차지합니다. C 언어에서 int는 4바이트를 차지하며 이는 전체적으로 20배의 차이입니다.

하지만 인터넷에서는 memory_get_usage()가 반환한 결과가 모두 배열 점유가 아니라 PHP 자체의 일부 구조를 포함한다고 합니다. 따라서 다른 방법을 시도하고 PHP 내장 함수를 사용하여 배열을 생성하세요.


<?php 
  $start = memory_get_usage(); 
  $a = array_fill(0, 10000, 1); 
  $end = memory_get_usage(); //10k elements array; 
  echo &#39;argv:&#39;, ($end - $start )/10000,&#39;byte&#39; , &#39;<br>&#39;;
로그인 후 복사

출력은 다음과 같습니다.

argv:54.5792byte

는 이전보다 약간 나아졌지만 여전히 54바이트로 실제로 약 10배 더 나쁩니다.

이유는 PHP의 기본 구현에서 시작되어야 합니다. PHP는 int, double, string 등에 관계없이 약한 유형의 언어이므로 통합된 '$'로 모든 문제를 해결할 수 있습니다. PHP의 최하위 계층은 C 언어로 구현됩니다. 각 변수는 zval 구조에 해당하며 자세한 정의는 다음과 같습니다.


typedef struct _zval_struct zval; 
struct _zval_struct { 
  /* Variable information */ 
  zvalue_value value;   /* The value 1 12字节(32位机是12,64位机需要8+4+4=16) */ 
  zend_uint refcount__gc; /* The number of references to this value (for GC) 4字节 */ 
  zend_uchar type;    /* The active type 1字节*/ 
  zend_uchar is_ref__gc; /* Whether this value is a reference (&) 1字节*/ 
};
로그인 후 복사

PHP는 변수 값을 저장하기 위해 통합 구조를 사용합니다. zval은 다음과 같이 정의된 공용체입니다.


typedef union _zvalue_value { 
  long lval;         /* long value */ 
  double dval;        /* double value */ 
  struct {          /* string value */ 
    char *val; 
    int len; 
  } str;  
  HashTable *ht;       /* hash table value */ 
  zend_object_value obj;   /*object value */ 
} zvalue_value;
로그인 후 복사

공용체 유형이 차지하는 메모리 크기는 가장 큰 멤버가 차지하는 데이터 공간에 따라 결정됩니다. zvalue_value에서는 str 구조체의 int가 4바이트를 차지하고 char 포인터가 4바이트를 차지하므로 전체 zvalue_value가 차지하는 메모리는 8바이트이다.

zval의 크기는 8 + 4 + 1 + 1 = 14바이트입니다.

zvalue_value에도 HashTable이 있다는 점에 주목하세요. zval에서는 배열, 문자열 및 객체에도 추가 저장 구조가 필요합니다. 배열의 저장 구조는 HashTable입니다.

HashTable 정의는 다음을 제공합니다.


typedef struct _hashtable { 
   uint nTableSize; //表长度,并非元素个数 
   uint nTableMask;//表的掩码,始终等于nTableSize-1 
   uint nNumOfElements;//存储的元素个数 
   ulong nNextFreeElement;//指向下一个空的元素位置 
   Bucket *pInternalPointer;//foreach循环时,用来记录当前遍历到的元素位置 
   Bucket *pListHead; 
   Bucket *pListTail; 
   Bucket **arBuckets;//存储的元素数组 
   dtor_func_t pDestructor;//析构函数 
   zend_bool persistent;//是否持久保存。从这可以发现,PHP数组是可以实现持久保存在内存中的,而无需每次请求都重新加载。 
   unsigned char nApplyCount; 
   zend_bool bApplyProtection; 
} HashTable;
로그인 후 복사

테이블의 크기와 포함된 요소 수를 기록하는 여러 속성 변수 외에도 Bucket은 여러 번 사용됩니다. Bucket이 정의되는 방법:


typedef struct bucket { 
   ulong h; //数组索引 
   uint nKeyLength; //字符串索引的长度 
   void *pData; //实际数据的存储地址 
   void *pDataPtr; //引入的数据存储地址 
   struct bucket *pListNext; 
   struct bucket *pListLast; 
   struct bucket *pNext; //双向链表的下一个元素的地址 
   struct bucket *pLast;//双向链表的下一个元素地址 
   char arKey[1]; /* Must be last element */ 
} Bucket;
로그인 후 복사

Linked List와 마찬가지로 Bucket은 특정 데이터와 포인터가 있는 Linked List 노드인 반면, HashTable은 Bucket 요소의 문자열을 저장하는 배열입니다. PHP에서 다차원 배열의 구현은 버킷에 저장된 또 다른 HashTable입니다.

HashTable이 39바이트, Bucket이 33바이트를 차지한다고 계산해 보세요. 빈 배열은 14 + 39 + 33 = 86바이트를 차지합니다. Bucket 구조는 33바이트가 필요하며 키 길이가 4바이트보다 긴 부분은 Bucket에 추가되며 요소 값은 zval 구조일 가능성이 높습니다. 또한 각 배열에는 다음을 가리키는 Bucket 포인터 배열이 할당됩니다. arBuckets에서, 말할 수는 없지만 각 추가 요소에는 포인터가 필요하지만 상황은 더 나쁠 수 있습니다. 이는 하나의 배열 요소가 54바이트를 차지할 것이라고 계산하는데, 이는 위의 추정치와 거의 동일합니다.

공간 관점에서 볼 때 작은 배열은 평균적으로 비용이 많이 듭니다. 물론 스크립트는 많은 수의 작은 배열로 채워지지 않으며 더 작은 공간 비용으로 프로그래밍 속도를 얻을 수 있습니다. 그러나 배열을 컨테이너로 사용하면 이야기가 달라집니다. 실제 응용 프로그램에서는 요소가 많은 다차원 배열을 자주 접하게 됩니다. 예를 들어, 10k 요소의 1차원 배열은 약 540k의 메모리를 소비하는 반면 10k의 2차원 배열은 실제로 23M을 소비합니다. 작은 배열은 실제로 그럴 가치가 없습니다.

PHP 배열의 메모리 활용률이 낮은 이유에 대해 여기서 이야기해 보겠습니다. 다음 기사에서는 PHP 배열 작업의 구체적인 구현을 설명합니다.

관련 권장 사항:

PHP 배열이 메모리를 너무 많이 소비하는 문제에 대한 솔루션_PHP 튜토리얼

PHP 배열이 메모리를 너무 많이 소비하는 문제에 대한 솔루션

PHP 배열도 메모리를 소비하는 문제에 대한 솔루션 많은 메모리 Method_php 팁

위 내용은 PHP 배열의 낮은 메모리 활용도에 대한 자세한 분석의 상세 내용입니다. 자세한 내용은 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 옷 제거제

AI Hentai Generator

AI Hentai Generator

AI Hentai를 무료로 생성하십시오.

뜨거운 도구

메모장++7.3.1

메모장++7.3.1

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

SublimeText3 중국어 버전

SublimeText3 중국어 버전

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

스튜디오 13.0.1 보내기

스튜디오 13.0.1 보내기

강력한 PHP 통합 개발 환경

드림위버 CS6

드림위버 CS6

시각적 웹 개발 도구

SublimeText3 Mac 버전

SublimeText3 Mac 버전

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

CakePHP 프로젝트 구성 CakePHP 프로젝트 구성 Sep 10, 2024 pm 05:25 PM

이번 장에서는 CakePHP의 환경 변수, 일반 구성, 데이터베이스 구성, 이메일 구성에 대해 알아봅니다.

Ubuntu 및 Debian용 PHP 8.4 설치 및 업그레이드 가이드 Ubuntu 및 Debian용 PHP 8.4 설치 및 업그레이드 가이드 Dec 24, 2024 pm 04:42 PM

PHP 8.4는 상당한 양의 기능 중단 및 제거를 통해 몇 가지 새로운 기능, 보안 개선 및 성능 개선을 제공합니다. 이 가이드에서는 Ubuntu, Debian 또는 해당 파생 제품에서 PHP 8.4를 설치하거나 PHP 8.4로 업그레이드하는 방법을 설명합니다.

CakePHP 날짜 및 시간 CakePHP 날짜 및 시간 Sep 10, 2024 pm 05:27 PM

cakephp4에서 날짜와 시간을 다루기 위해 사용 가능한 FrozenTime 클래스를 활용하겠습니다.

소식통에 따르면 삼성전자와 SK하이닉스는 2026년 이후 적층형 모바일 메모리를 상용화할 것으로 보인다. 소식통에 따르면 삼성전자와 SK하이닉스는 2026년 이후 적층형 모바일 메모리를 상용화할 것으로 보인다. Sep 03, 2024 pm 02:15 PM

3일 홈페이지 보도에 따르면 국내 언론 에트뉴스는 어제(현지시간) 삼성전자와 SK하이닉스의 'HBM형' 적층구조 모바일 메모리 제품이 2026년 이후 상용화될 것이라고 보도했다. 소식통에 따르면 두 한국 메모리 거대 기업은 적층형 모바일 메모리를 미래 수익의 중요한 원천으로 여기고 'HBM형 메모리'를 스마트폰, 태블릿, 노트북으로 확장해 엔드사이드 AI에 전력을 공급할 계획이라고 전했다. 이 사이트의 이전 보도에 따르면 삼성전자 제품은 LPWide I/O 메모리라고 하며 SK하이닉스는 이 기술을 VFO라고 부른다. 두 회사는 팬아웃 패키징과 수직 채널을 결합하는 것과 거의 동일한 기술 경로를 사용했습니다. 삼성전자 LPWide I/O 메모리의 비트폭은 512이다.

CakePHP 파일 업로드 CakePHP 파일 업로드 Sep 10, 2024 pm 05:27 PM

파일 업로드 작업을 위해 양식 도우미를 사용할 것입니다. 다음은 파일 업로드의 예입니다.

CakePHP 라우팅 CakePHP 라우팅 Sep 10, 2024 pm 05:25 PM

이번 장에서는 라우팅과 관련된 다음과 같은 주제를 학습하겠습니다.

CakePHP 토론 CakePHP 토론 Sep 10, 2024 pm 05:28 PM

CakePHP는 PHP용 오픈 소스 프레임워크입니다. 이는 애플리케이션을 훨씬 쉽게 개발, 배포 및 유지 관리할 수 있도록 하기 위한 것입니다. CakePHP는 강력하고 이해하기 쉬운 MVC와 유사한 아키텍처를 기반으로 합니다. 모델, 뷰 및 컨트롤러 gu

CakePHP 유효성 검사기 만들기 CakePHP 유효성 검사기 만들기 Sep 10, 2024 pm 05:26 PM

컨트롤러에 다음 두 줄을 추가하면 유효성 검사기를 만들 수 있습니다.

See all articles