PHP에서 Copy On Write의 코드 예제 공유

黄舟
풀어 주다: 2023-03-06 21:44:01
원래의
2407명이 탐색했습니다.

문제 소개

먼저 의 과제와 참고문제

<?php$a = 10;//将常量值赋给变量,会为a分配内存空间 
$b = $a;//变量赋值给变量,是不是copy了一份副本,b也分配了内存空间呢? 
$c = &$a;//引用是不会为c分配空间的,c和a是共用一份空间的。?>
로그인 후 복사
로그인 후 복사
를 살펴보겠습니다. PHP

중간 질문에 대한 귀하의 답변은 무엇입니까? 오늘 이전에 내 대답은 b에 메모리 공간이 할당된다는 것이었습니다. 제가 이해하는 바는 이것이기 때문입니다.
&가 할당되면 별칭을 정의하는 변수로 간주되어 메모리 공간에 대한 참조를 추가합니다. 그 중 하나를 변경하면 다른 참조에 영향을 미칩니다. unset()을 사용하면 가변 메모리 공간에 대한 참조만 끊어지고, 메모리 공간은 해제되지 않습니다.
= 할당은 다르며, 원래 변수의 복사본을 저장하기 위해 메모리 공간을 다시 엽니다. 둘 사이의 수정은 서로 영향을 미치지 않습니다.

다음 프로그램은 이를 확인합니다.

<?php$a = 10;   //将常量值赋给变量,会为a分配内存空间 
$b = $a; //变量赋值给变量,是不是copy了一份副本,b也分配了内存空间呢? 
$c = &$a; //引用是不会为c分配空间的,c和a是共用一份空间的。 
$a = 5;echo $c;   //输出5,因为a和c 是指向同一个内存空间echo PHP_EOL;
echo $b;   //由于b是副本,对a的操作不会影响b,输出10?>
로그인 후 복사

그러면

$b = $a;//之后a  和  b 都不做任何改变,保持一致
로그인 후 복사
로그인 후 복사

그런 문제가 있는 경우 if = 할당 후에 두 변수 모두 변경되지 않았습니다. 복사본이 두 개인 경우 메모리 낭비가 발생합니까?
실제로 PHP에서는 이러한 상황을 피할 수 있습니다.
PHP에서 새 변수에 변수를 할당하면 새 변수에 대해 메모리 공간이 즉시 할당되지 않고 메모리 공간에 대한 참조가 추가됩니다. 원래 변수나 새 변수가 변경되면 새 변수에 메모리 공간이 할당됩니다.

<?php$a = 1;$b = $a; 
echo $a;//在此之前,b都是和a共用内存空间的。 $a = 2;//a作出了改变,此时b才会有自己的空间?>
로그인 후 복사
로그인 후 복사

php 변수는 "zval"이라는 변수 컨테이너에 저장됩니다. zval 변수 컨테이너에는 변수의 유형과 값 외에 2바이트의 추가 정보가 포함되어 있습니다. 첫 번째는 "is_ref"로, 이 변수가 참조 세트(referenceset)에 속하는지 식별하는 데 사용되는 부울 값입니다. 이 바이트를 통해 PHP 엔진은 일반 변수와 참조 변수를 구별할 수 있습니다. PHP에서는 사용자가 &를 사용하여 사용자 정의 참조를 사용할 수 있으므로 zval 변수 컨테이너에는 메모리 사용을 최적화하는 내부 참조 계산 메커니즘도 있습니다. 두 번째 추가 바이트는 "refcount"이며, 이 zval 변수 컨테이너를 가리키는 변수(기호라고도 함)의 수를 나타내는 데 사용됩니다. "refcount" 값이 1이면 "is_ref" 값은 항상 FALSE입니다.

xdebug를 설치한 후 xdebug_debug_zval()을 사용하면 zval 구조를 볼 수 있습니다.
다음과 같습니다.

<?php
$a = 1;
$b = $a;
echo $a;//在此之前,b都是和a共用内存空间的。xdebug_debug_zval(&#39;b&#39;);
$a = 2;//a作出了改变,此时b才会有自己的空间xdebug_debug_zval(&#39;b&#39;);?>
로그인 후 복사
로그인 후 복사

출력:

b: 
  (refcount=2, is_ref=0), 
  int  
  1
b: 
  (refcount=1, is_ref=0), 
  int  
  1
로그인 후 복사
로그인 후 복사

위 결과에서 a가 변경되기 전의 참조 카운트는 2이고, a가 변경된 후에는 b의 참조 카운트가 1이 되는 것을 알 수 있습니다. .b가 공간을 재할당하기 때문입니다.
위에서 설명한 현상은 Copy-On-Write입니다.

Copy-on-Write

 Copy-on-Write, 줄여서 COW라고도 함) 즉, 쓰기 시 수정을 위해 실제로 메모리 복사본을 복사한다는 것입니다. COW는 스레드와 메모리 사용을 최적화하기 위해 *nix 시스템에서 처음 사용되었으며 이후 C++의 STL 등 다양한 프로그래밍 언어에서 널리 사용되었습니다. PHP 커널에서 COW는 주요 메모리 최적화 방법이기도 합니다. 변수와 메모리에 대한 이전 논의에서 참조 카운팅은 변수의 파괴와 재활용을 식별하는 데 중요한 역할을 합니다. 참조 카운팅의 목적은 COW가 정상적으로 작동할 수 있도록 하여 최적의 메모리 사용을 달성하는 것입니다.   기록 중 복사의 장점
: 변수에 값을 할당할 때 새 변수가 저장한 값을 저장하기 위해 새 메모리를 적용하지 않고 단순히 메모리를 공유합니다. 참조 중 하나가 가리키는 변수의 값이 변경되는 경우에만 메모리 사용량을 줄이기 위해 값 내용을 저장하기 위해 새로운 공간이 할당됩니다.  

PHP의 기본 데이터 구조 관점에서

ref_count 및 is_ref는 zval
구조

에 정의되어 있습니다. > is_ref 식별자는 사용자가 &를 필수 참조로 사용하는지 여부입니다. Ref_count는 이 zval이 얼마나 많은 변수를 참조하는지 식별하는 데 사용되는 참조 횟수입니다. 즉, 쓰기 시 자동으로 복사되는 참조입니다.

问题引入

  首先来看看PHP中的赋值与引用问题

<?php$a = 10;//将常量值赋给变量,会为a分配内存空间 
$b = $a;//变量赋值给变量,是不是copy了一份副本,b也分配了内存空间呢? 
$c = &$a;//引用是不会为c分配空间的,c和a是共用一份空间的。?>
로그인 후 복사
로그인 후 복사

  对于中间的那个问题,你的答案是什么呢?在今天之前,我的答案是会为b分配内存空间。因为我是这么理解的:
  &赋值的时候,视为一个变量定义了一个别名,增加了一个对内存空间的引用。改变其中一个,会影响其他的引用。而使用unset()时,只是断开了对变量内存空间的引用,内存空间不会释放。
  而 = 赋值则不同,它会重新开辟一份内存空间存储原变量的副本。两者之间的修改不会相互影响。

  而下面的程序则印证了这一点:

<?php
$a = 10;   //将常量值赋给变量,会为a分配内存空间 
$b = $a; //变量赋值给变量,是不是copy了一份副本,b也分配了内存空间呢? 
$c = &$a; //引用是不会为c分配空间的,c和a是共用一份空间的。 
$a = 5;echo $c;   //输出5,因为a和c 是指向同一个内存空间echo PHP_EOL;
echo $b;   //由于b是副本,对a的操作不会影响b,输出10?>
로그인 후 복사

  那如果

$b = $a;//之后a  和  b 都不做任何改变,保持一致
로그인 후 복사
로그인 후 복사

  有这么一个问题,如果 = 赋值之后,两个变量都不曾改变,如果是两份副本,岂不是太浪费内存?
  PHP中实际上避免了这种情况。
  PHP中将一个变量赋值给新变量时,不会立即为新变量分配内存空间,只是增加了对内存空间的引用。当原变量或者新变量作出任何改变时,才会为新变量 分配一块内存空间。

<?php$a = 1;$b = $a; 
echo $a;//在此之前,b都是和a共用内存空间的。 $a = 2;//a作出了改变,此时b才会有自己的空间?>
로그인 후 복사
로그인 후 복사

  每个php变量存在一个叫”zval”的变量容器中。一个zval变量容器,除了包含变量的类型和值,还包括两个字节的额外信息。第一个是”is_ref”,是个bool值,用来标识这个变量是否是属于引用集合(referenceset)。通过这个字节,php引擎才能把普通变量和引用变量区分开来,由于php允许用户通过使用&来使用自定义引用,zval变量容器中还有一个内部引用计数机制,来优化内存使用。第二个额外字节是”refcount”,用以表示指向这个zval变量容器的变量(也称符号即symbol)个数。当”refcount”的值是1时,”is_ref”的值总是FALSE.

  安装xdebug之后,利用xdebug_debug_zval(),可以看到zval结构:
  如下:

<?php
$a = 1;
$b = $a;
echo $a;//在此之前,b都是和a共用内存空间的。xdebug_debug_zval(&#39;b&#39;);
$a = 2;//a作出了改变,此时b才会有自己的空间xdebug_debug_zval(&#39;b&#39;);?>
로그인 후 복사
로그인 후 복사

 输出:

b: 
  (refcount=2, is_ref=0), 
  int  
  1
b: 
  (refcount=1, is_ref=0), 
  int  
  1
로그인 후 복사
로그인 후 복사

  由上面的结果可以看到,在a作出改变之前,引用计数是2 ,当a作出改变之后,b的引用计数变为1,是因为b重新分配了空间。
  上面说描述的现象就是写时复制。

写时复制

  写时复制Copy-on-Write,也缩写为COW),顾名思义,就是在写入时才真正复制一份内存进行修改。 COW最早应用在*nix系统中对线程与内存使用的优化,后面广泛的被使用在各种编程语言中,如C++的STL等。 在PHP内核中,COW也是主要的内存优化手段。 在前面关于变量和内存的讨论中,引用计数对变量的销毁与回收中起着至关重要的标识作用。 引用计数存在的意义,就是为了使得COW可以正常运作,从而实现对内存的优化使用。
   写时复制优点:是通过赋值的方式赋值给变量时不会申请新内存来存放新变量所保存的值,而是简单的通过一个计数器来共用内存,只有在其中的一个引用指向变量的值发生变化时才申请新空间来保存值内容以减少对内存的占用。

  从PHP底层基础数据结构来看

ref_count和is_ref是定义于zval结构体中;
  is_ref标识是不是用户使用 & 的强制引用;
 ref_count是引用计数,用于标识此zval被多少个变量引用,即写时复制的自动引用,为0时会被销毁。

위 내용은 PHP에서 Copy On Write의 코드 예제 공유의 상세 내용입니다. 자세한 내용은 PHP 중국어 웹사이트의 기타 관련 기사를 참조하세요!

관련 라벨:
원천:php.cn
본 웹사이트의 성명
본 글의 내용은 네티즌들의 자발적인 기여로 작성되었으며, 저작권은 원저작자에게 있습니다. 본 사이트는 이에 상응하는 법적 책임을 지지 않습니다. 표절이나 침해가 의심되는 콘텐츠를 발견한 경우 admin@php.cn으로 문의하세요.
인기 튜토리얼
더>
최신 다운로드
더>
웹 효과
웹사이트 소스 코드
웹사이트 자료
프론트엔드 템플릿