이 글은 PHP 변수의 참조 할당 및 값 할당에 대한 자세한 소개(코드)를 제공합니다. 특정 참조 값이 있으므로 도움이 될 수 있습니다.
// 定义一个变量 $a = range(0, 10000); var_dump(memory_get_usage()); // 定义变量b,将a变量的值赋值给b $b = $a; var_dump(memory_get_usage()); // 对a进行修改 // COW: Copy-On-Write $a = range(0, 10000); var_dump(memory_get_usage());
출력 결과:
int(989768) int(989856) int(1855608)
$a = range(0, 10000);
$a = range(0, 10000);
$b = $a;
$a = range(0, 10000);
PHP写时复制机制(Copy-on-Write,也缩写为COW)
顾名思义,就是在写入时才真正复制一份内存进行修改。
COW最早应用在Unix系统中对线程与内存使用的优化,后面广泛的被使用在各种编程语言中,如C++的STL等。
在PHP内核中,COW也是主要的内存优化手段。
在通过变量赋值的方式赋值给变量时,不会申请新内存来存放新变量的值,而是简单的通过一个计数器来共用内存。只有在其中的一个引用指向变量的值发生变化时,才申请新空间来保存值内容,以减少对内存的占用。
在很多场景下PHP都使用COW进行内存的优化。比如:变量的多次赋值、函数参数传递,并在函数体内修改实参等。
// 定义一个变量 $a = range(0, 10000); var_dump(memory_get_usage()); // 定义变量b,将a变量的引用赋给b $b = &$a; var_dump(memory_get_usage()); // 对a进行修改 $a = range(0, 10000); var_dump(memory_get_usage());
输出结果:
int(989760) int(989848) int(989840)
$a = range(0, 10000);
$b = &$a;
$a = range(0, 10000);
xdebug_debug_zval()
查看变量的引用情况xdebug_debug_zval()
用于显示变量的信息。需要安装xdebug扩展。
$a = 1; xdebug_debug_zval('a'); // 定义变量b,把a的值赋值给b $b = $a; xdebug_debug_zval('a'); xdebug_debug_zval('b'); // a进行写操作 $a = 2; xdebug_debug_zval('a'); xdebug_debug_zval('b');
输出结果:
a: (refcount=1, is_ref=0)=1 a: (refcount=2, is_ref=0)=1 b: (refcount=2, is_ref=0)=1 a: (refcount=1, is_ref=0)=2 b: (refcount=1, is_ref=0)=1
$a = 1;
$a = 1; xdebug_debug_zval('a');
输出
a: (refcount=1, is_ref=0)=1
refcount=1
表示该变量指向的内存地址的引用个数变为1is_ref=0
表示该变量不是引用
$b
,把 $a
的值赋给 $b
, $b = $a;
$b = $a; xdebug_debug_zval('a'); xdebug_debug_zval('b');
输出
a: (refcount=2, is_ref=0)=1 b: (refcount=2, is_ref=0)=1
refcount=2
表示该变量指向的内存地址的引用个数变为2is_ref=0
表示该变量不是引用
$a
进行写操作 $a = 2;
$a = 2; xdebug_debug_zval('a'); xdebug_debug_zval('b');
输出
a: (refcount=1, is_ref=0)=2 b: (refcount=1, is_ref=0)=1
因为COW机制,对变量 $a
进行写操作时,会为变量 $a
新分配一块内存空间,用于存储变量 $a
的值。
此时 $a
和 $b
指向的内存地址的引用个数都变为1。
$a = 1; xdebug_debug_zval('a'); // 定义变量b,把a的引用赋给b $b = &$a; xdebug_debug_zval('a'); xdebug_debug_zval('b'); // a进行写操作 $a = 2; xdebug_debug_zval('a'); xdebug_debug_zval('b');
a: (refcount=1, is_ref=0)=1 a: (refcount=2, is_ref=1)=1 b: (refcount=2, is_ref=1)=1 a: (refcount=2, is_ref=1)=2 b: (refcount=2, is_ref=1)=2
$a = 1;
$a = 1; xdebug_debug_zval('a');
输出
a: (refcount=1, is_ref=0)=1
refcount=1
表示该变量指向的内存地址的引用个数变为1is_ref=0
$b = $a;
$a 수정 = 범위( 0 , 10000);
🎜🎜🎜🎜🎜PHP 쓰기 중 복사 메커니즘(Copy-on-Write, 줄여서 COW)🎜🎜이름에서 알 수 있듯이 복사본은 실제로 메모리를 쓸 때 만들어집니다. $b = &$a; xdebug_debug_zval('a'); xdebug_debug_zval('b');🎜출력 결과: 🎜
a: (refcount=2, is_ref=1)=1 b: (refcount=2, is_ref=1)=1🎜🎜변수 정의
$a = range(0, 10000);
🎜🎜🎜🎜🎜 🎜정의 변수 b, 변수 a의 참조를 b $b = &$a;
🎜🎜🎜🎜🎜🎜$a = range(0, 10000) ;
🎜🎜🎜🎜🎜2. 변수의 참조 상태를 보려면 xdebug_debug_zval()
을 사용하세요🎜🎜xdebug_debug_zval()
이 사용됩니다 변수 정보를 표시합니다. xdebug 확장을 설치해야 합니다. 🎜🎜1. 값별 할당🎜$a = 2; xdebug_debug_zval('a'); xdebug_debug_zval('b');🎜출력 결과: 🎜
a: (refcount=2, is_ref=1)=2 b: (refcount=2, is_ref=1)=2🎜🎜변수 정의
$a = 1;
$a = 1; $b = &$a; // unset 只会取消引用,不会销毁内存空间 unset($b); echo $a;🎜Output🎜
1🎜
refcount= 1
은 변수가 가리키는 메모리 주소에 대한 참조 개수가 1이 된다는 의미is_ref=0
는 변수가 참조가 아니라는 의미🎜🎜🎜 🎜🎜 🎜🎜정의 변수$b
, $a
값을 $b
에 할당, $b = $a;
$a = 1; $b = &$a;🎜Output🎜
unset($b);🎜
refcount=2
는 변수가 가리키는 메모리 주소에 대한 참조 개수가 2개가 된다는 뜻입니다is_ref =0
은 변수가 참조가 아님을 의미합니다🎜 🎜🎜 🎜🎜🎜🎜 변수 $a
$a = 2;
에 쓰기 >echo $a;🎜Output🎜
1🎜COW 메커니즘으로 인해 변수
$a
가 쓰기 작업을 수행하면 $a
변수에 대해 새로운 메모리 공간이 할당됩니다. 변수 $a
의 값을 저장합니다. $a
와 $b
가 가리키는 메모리 주소에 대한 참조 개수는 모두 1이 됩니다. 🎜🎜🎜🎜🎜🎜2. 참조 할당🎜class Person { public $age = 1; } $p1 = new Person; xdebug_debug_zval('p1'); $p2 = $p1; xdebug_debug_zval('p1'); xdebug_debug_zval('p2'); $p2->age = 2; xdebug_debug_zval('p1'); xdebug_debug_zval('p2');
p1: (refcount=1, is_ref=0)=class Person { public $age = (refcount=2, is_ref=0)=1 } p1: (refcount=2, is_ref=0)=class Person { public $age = (refcount=2, is_ref=0)=1 } p2: (refcount=2, is_ref=0)=class Person { public $age = (refcount=2, is_ref=0)=1 } p1: (refcount=2, is_ref=0)=class Person { public $age = (refcount=1, is_ref=0)=2 } p2: (refcount=2, is_ref=0)=class Person { public $age = (refcount=1, is_ref=0)=2 }🎜🎜변수 정의
$a = 1;
$p1 = new Person; xdebug_debug_zval('p1');🎜Output🎜
p1: (refcount=1, is_ref=0)=class Person { public $age = (refcount=2, is_ref=0)=1 }🎜
refcount=1
변수가 가리키는 메모리 주소에 대한 참조 개수가 1개가 됨을 나타냅니다.is_ref=0
변수가 참조가 아님을 나타냅니다🎜🎜🎜🎜🎜🎜$b
,把 $a
的引用赋给 $b
, $b = &$a;
$b = &$a; xdebug_debug_zval('a'); xdebug_debug_zval('b');
输出
a: (refcount=2, is_ref=1)=1 b: (refcount=2, is_ref=1)=1
refcount=2
表示该变量指向的内存地址的引用个数变为2is_ref=1
表示该变量是引用
$a
进行写操作 $a = 2;
$a = 2; xdebug_debug_zval('a'); xdebug_debug_zval('b');
输出
a: (refcount=2, is_ref=1)=2 b: (refcount=2, is_ref=1)=2
因为变量 $a
和变量 $b
指向相同的内存地址,其实引用。
对变量 $a
进行写操作时,会直接修改指向的内存空间的值,因此变量 $b
的值会跟着一起改变。
$a = 1; $b = &$a; // unset 只会取消引用,不会销毁内存空间 unset($b); echo $a;
输出
1
$a
,并将 $a
的引用赋给变量 $b
$a = 1; $b = &$a;
$b
unset($b);
$a
虽然销毁的 $b
,但是 $a
的引用和内存空间依旧存在。
echo $a;
输出
1
class Person { public $age = 1; } $p1 = new Person; xdebug_debug_zval('p1'); $p2 = $p1; xdebug_debug_zval('p1'); xdebug_debug_zval('p2'); $p2->age = 2; xdebug_debug_zval('p1'); xdebug_debug_zval('p2');
p1: (refcount=1, is_ref=0)=class Person { public $age = (refcount=2, is_ref=0)=1 } p1: (refcount=2, is_ref=0)=class Person { public $age = (refcount=2, is_ref=0)=1 } p2: (refcount=2, is_ref=0)=class Person { public $age = (refcount=2, is_ref=0)=1 } p1: (refcount=2, is_ref=0)=class Person { public $age = (refcount=1, is_ref=0)=2 } p2: (refcount=2, is_ref=0)=class Person { public $age = (refcount=1, is_ref=0)=2 }
$p1 = new Person;
$p1 = new Person; xdebug_debug_zval('p1');
输出
p1: (refcount=1, is_ref=0)=class Person { public $age = (refcount=2, is_ref=0)=1 }
refcount=1
表示该变量指向的内存地址的引用个数变为1is_ref=0
表示该变量不是引用
$p1
赋给 $p2
$p2 = $p1; xdebug_debug_zval('p1'); xdebug_debug_zval('p2');
输出
p1: (refcount=2, is_ref=0)=class Person { public $age = (refcount=2, is_ref=0)=1 } p2: (refcount=2, is_ref=0)=class Person { public $age = (refcount=2, is_ref=0)=1 }
refcount=2
表示该变量指向的内存地址的引用个数变为2
$p2
中的属性 age
进行写操作$p2->age = 2; xdebug_debug_zval('p1'); xdebug_debug_zval('p2');
输出
p1: (refcount=2, is_ref=0)=class Person { public $age = (refcount=1, is_ref=0)=2 } p2: (refcount=2, is_ref=0)=class Person { public $age = (refcount=1, is_ref=0)=2 }
因为php中对象本身就是引用赋值。对 $p2
中的属性 age
进行写操作时,会直接修改指向的内存空间的值,因此变量 $p1
的 age
属性的值会跟着一起改变。
/** * 写出如下程序的输出结果 * * $d = ['a', 'b', 'c']; * * foreach($d as $k => $v) * { * $v = &$d[$k]; * } * * 程序运行时,每一次循环结束后变量 $d 的值是什么?请解释。 * 程序执行完成后,变量 $d 的值是什么?请解释。 */
foreach
时 $v
、$d[$k]
的值$k = 0 $v = 'a' $d[$k] = $d[0] = 'a'
此时,$v
和 $d[0]
在内存中分别开辟了一块空间
![$v 和 $d[0] 在内存中分别开辟了一块空间](http://md.ws65535.top/xsj/201...
$v = &$d[0]
改变了 $v 指向的内存地址$v = &$d[0]
![$v = &$d[0] 改变了 $val 指向的内存地址](http://md.ws65535.top/xsj/201...
['a', 'b', 'c']
foreach
时 $v
被赋值为 'b',此时$v
指向的内存地址与 $d[0]
相同,且为引用,因此 $d[0]
的值被修改为 'b'$v = 'b'
=> $d[0] = 'b'
![$v = ‘b’ => $d[0] = ‘b’](http://md.ws65535.top/xsj/201...
foreach
时 $d[$k]
的值$k = 1 $d[$k] = $d[1] = 'b'
![$d[2] = ‘b’](http://md.ws65535.top/xsj/201...
$v = &$d[1]
改变了 $v 指向的内存地址$v = &$d[1]
![$v = &$d[1]](http://md.ws65535.top/xsj/201...
$d
的值['b', 'b', 'c']
foreach
时 $v
被赋值为 'c',此时$v
指向的内存地址与 $d[1]
相同,且为引用,因此 $d[1]
的值被修改为 'c'$v = 'c'
=> $d[1] = 'c'
![$v = ‘c’ => $d[1] = ‘c’](http://md.ws65535.top/xsj/201...
foreach
时 $d[$k]
的值$k = 2 $d[2] = 'c'
![$d[2] = ‘c’](http://md.ws65535.top/xsj/201...
$v = &$d[2]
改变了 $v 指向的内存地址$v = &$d[2]
![$v = &$d[2]](http://md.ws65535.top/xsj/201...
$d
的值['b', 'c', 'c']
$d = ['a', 'b', 'c']; foreach ($d as $k=>$v) { $v = &$d[$k]; print_r($d); } print_r($d);
Array ( [0] => a [1] => b [2] => c ) Array ( [0] => b [1] => b [2] => c ) Array ( [0] => b [1] => c [2] => c ) Array ( [0] => b [1] => c [2] => c )
相关推荐:
PHP变量赋值、代入给JavaScript中的变量,赋值javascript
위 내용은 PHP 변수(코드)의 참조 할당 및 값 할당에 대한 자세한 소개의 상세 내용입니다. 자세한 내용은 PHP 중국어 웹사이트의 기타 관련 기사를 참조하세요!