PHP赋值语句的引用传递与按值传递

2024-08-21 20:38

本文主要是介绍PHP赋值语句的引用传递与按值传递,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

赋值语句的引用传递与按值传递

之前对于php当中的引用传递与按值传递,并没有深入了解。

php中对变量赋值有2种方式:

  1. 按值传递
  2. 引用传递

这2个主要应用于变量或者函数赋值,计数赋值用于对象赋值。

按值传递

简单来说就是将变量进行复制,然后开辟一块新的内存空间存储。比如:

      $b = 1;$a = $b;$a = 2;dd($a,$b);

结果为 2 1

$a的改变并不会影响$b的数值。变量相当于一个内存中的地址别名,当$a = $b,就开辟一个新的内存空间,将 a 地 址 内 存 的 数 据 取 出 来 放 在 a地址内存的数据取出来放在 ab所指向的地址中。因为是不同的地址内存中,所以互相不会影响。按值赋值php7之前有个copy on write(写时复制)机制,貌似到php7废除了(以待以后考证)。

引用传递

引用传递则是拷贝一份变量的引用。关于变量的引用,有一篇解释的非常好的文章http://www.cnblogs.com/mushan/p/4330386.html。简单来说,php中的变量与值是2个不同的概念,变量存于一个符号表中,并有作用域,而值则在php内部(zend引擎中),存于一个zval结构体当中。

is_ref 代表他是否为引用,是一个bool值,用来标识这个值是否是一个引用。第二个额外字节是refcount,用来表示指向这个值的变量(也称符号即symbol)的个数。如果refcount为0,那么这个值就可以被回收了。

测试代码如下:

      $b = 1;$a = $b;$a = 2;xdebug_debug_zval('a');xdebug_debug_zval('b');$c = 1;$d = &$c;$d = 2;xdebug_debug_zval('c');xdebug_debug_zval('d');dd($a,$b);

测试结果:

a:(refcount=0, is_ref=0)int 2
b:(refcount=0, is_ref=0)int 1
c:(refcount=2, is_ref=1)int 2
d:(refcount=2, is_ref=1)int 22122

这里我的php版本为7.0.12,a b 变量的refcount都为0了,这是在php7当中对于在zval的value字段中能保存下的值,不在进行引用计数的原因。这里暂时不深入研究,不然又跑远了。
看一下c d 变量使用应用传递的时候,refcount数量都为2,有2个变量指向它,is_ref为1,这就说明c``d都指向了同一个变量,且变量被标记为引用值。当改变了d的的值,c也会随之改变,因为2者指向同一个zval容器。

默认传值方式

总结

搞清楚PHP引用赋值与按值赋值的过程当中,查了很多资料,也让自己由以前对2者模糊的印象,变得清晰。值得注意的是由于PHP版本不同,让自己测试当中遇到一些困难。PHP7数据结构发生了变化,性能也得到了提升。鸟哥的博客当中对这一块也有过解释,但是我自己的环境运行代码,与他得到的结果也不一样,和php手册上的也不一样。

总的来说,变量与数组一般都是按值赋值,对象默认是引用传递。当需要修改一个内存存储大量数据的变量,或者需要在函数内部修改变量,并返回修改的变量,使用引用传递(&)。

参考资料:

  • php手册:引用的解释 http://php.net/manual/zh/language.references.php

  • php手册:引用计数基本知识 http://php.net/manual/zh/features.gc.refcounting-basics.php

  • 鸟哥博客:深入理解PHP原理之变量分离/引用(Variables Separation) http://www.laruence.com/2008/09/19/520.html

  • php的三种变量赋值方式 http://blog.csdn.net/sanwenyublog/article/details/13021975

–END

这篇关于PHP赋值语句的引用传递与按值传递的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1094221

相关文章

Mybatis Plus JSqlParser解析sql语句及JSqlParser安装步骤

《MybatisPlusJSqlParser解析sql语句及JSqlParser安装步骤》JSqlParser是一个用于解析SQL语句的Java库,它可以将SQL语句解析为一个Java对象树,允许... 目录【一】jsqlParser 是什么【二】JSqlParser 的安装步骤【三】使用场景【1】sql语

sql语句字段截取方法

《sql语句字段截取方法》在MySQL中,使用SUBSTRING函数可以实现字段截取,下面给大家分享sql语句字段截取方法,感兴趣的朋友一起看看吧... 目录sql语句字段截取sql 截取表中指定字段sql语句字段截取1、在mysql中,使用SUBSTRING函数可以实现字段截取。例如,要截取一个字符串字

Java资源管理和引用体系的使用详解

《Java资源管理和引用体系的使用详解》:本文主要介绍Java资源管理和引用体系的使用,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录1、Java的引用体系1、强引用 (Strong Reference)2、软引用 (Soft Reference)3、弱引用 (W

Spring 中的循环引用问题解决方法

《Spring中的循环引用问题解决方法》:本文主要介绍Spring中的循环引用问题解决方法,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友参考下吧... 目录什么是循环引用?循环依赖三级缓存解决循环依赖二级缓存三级缓存本章来聊聊Spring 中的循环引用问题该如何解决。这里聊

Spring 请求之传递 JSON 数据的操作方法

《Spring请求之传递JSON数据的操作方法》JSON就是一种数据格式,有自己的格式和语法,使用文本表示一个对象或数组的信息,因此JSON本质是字符串,主要负责在不同的语言中数据传递和交换,这... 目录jsON 概念JSON 语法JSON 的语法JSON 的两种结构JSON 字符串和 Java 对象互转

C++使用printf语句实现进制转换的示例代码

《C++使用printf语句实现进制转换的示例代码》在C语言中,printf函数可以直接实现部分进制转换功能,通过格式说明符(formatspecifier)快速输出不同进制的数值,下面给大家分享C+... 目录一、printf 原生支持的进制转换1. 十进制、八进制、十六进制转换2. 显示进制前缀3. 指

MySQL 中的 LIMIT 语句及基本用法

《MySQL中的LIMIT语句及基本用法》LIMIT语句用于限制查询返回的行数,常用于分页查询或取部分数据,提高查询效率,:本文主要介绍MySQL中的LIMIT语句,需要的朋友可以参考下... 目录mysql 中的 LIMIT 语句1. LIMIT 语法2. LIMIT 基本用法(1) 获取前 N 行数据(

MySQL中动态生成SQL语句去掉所有字段的空格的操作方法

《MySQL中动态生成SQL语句去掉所有字段的空格的操作方法》在数据库管理过程中,我们常常会遇到需要对表中字段进行清洗和整理的情况,本文将详细介绍如何在MySQL中动态生成SQL语句来去掉所有字段的空... 目录在mysql中动态生成SQL语句去掉所有字段的空格准备工作原理分析动态生成SQL语句在MySQL

Go 语言中的select语句详解及工作原理

《Go语言中的select语句详解及工作原理》在Go语言中,select语句是用于处理多个通道(channel)操作的一种控制结构,它类似于switch语句,本文给大家介绍Go语言中的select语... 目录Go 语言中的 select 是做什么的基本功能语法工作原理示例示例 1:监听多个通道示例 2:带

mysql的基础语句和外键查询及其语句详解(推荐)

《mysql的基础语句和外键查询及其语句详解(推荐)》:本文主要介绍mysql的基础语句和外键查询及其语句详解(推荐),本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋... 目录一、mysql 基础语句1. 数据库操作 创建数据库2. 表操作 创建表3. CRUD 操作二、外键