目錄
您可能感兴趣的文章
首頁 php教程 php手册 php引用(&)详解及注意事项

php引用(&)详解及注意事项

Jun 21, 2016 am 08:46 AM
bar function print

PHP的引用(就是在变量函数对象对象方法 等前面加上&符号)

在PHP 中引用的意思是:不同的名字访问同一个变量内容。

与C语言中的指针是有差别的。C语言中的指针里面存储的是变量的内容,在内存中存放的地址。

1.变量的引用

PHP 的引用允许你用两个变量来指向同一个内容。

<?
$a = "ABC";
$b = &$a;
echo $a; //这里输出:ABC
echo $b; //这里输出:ABC
$b = "EFG";
echo $a; //这里$a的值变为EFG 所以输出EFG
echo $b; //这里输出EFG
?>
登入後複製

2.函数的引用传递(传址调用

传址调用我就不多说了,下面直接给出代码:

<?php
function test(&$a) {
	$a = $a + 100;
}
$b = 1;
echo $b; //输出1
test($b); //这里$b传递给函数的其实是$b的变量内容所处的内存地址,通过在函数里改变$a的值,就可以改变$b的值了。
echo "<br>";
echo $b; //输出101
?>
登入後複製

要注意的是,在这里 test(1); 的话就会出错,原因自己去想。

注意:

上面的“ test($b); ” 中的$b前面不要加 & 符号,但是在函数“call_user_func_array”中,若要引用传参,就得需要 & 符号,如下代码所示:

<?php
function a(&$b) {
	$b++;
}
$c = 0;
call_user_func_array('a', array(&$c));
echo $c; //输出 1
?>
登入後複製

3.函数的引用返回

先看代码:

<?php
function &test() {
	static $b = 0;//申明一个静态变量
	$b = $b + 1;
	echo $b;
	return $b;
}

$a = test(); //这条语句会输出 $b的值 为1
$a = 5;
$a = test(); //这条语句会输出 $b的值 为2

$a = &test(); //这条语句会输出 $b的值 为3
$a = 5;
$a = test(); //这条语句会输出 $b的值 为6
?>
登入後複製

下面解释下:

通过这种方式 $a=test(); 得到的其实不是函数的引用返回,这跟普通的函数调用没有区别,至于原因:这是PHP的规定。

PHP规定通过$a=&test(); 方式得到的才是函数的引用返回,至于什么是引用返回呢(PHP手册上说:引用返回用在当想用函数找到引用应该被绑定在哪一个变量上面时。) 这句狗屁话,害我半天没看懂。

用上面的例子来解释就是:

$a = test() 方式调用函数,只是将函数的值赋给$a而已,而$a做任何改变,都不会影响到函数中的$b,而通过 $a = &test() 方式调用函数呢,它的作用是将return $b中的$b变量的内存地址与$a变量的内存地址都指向了同一个地方,即产生了相当于这样的效果 ($a = &$b;) 所以改变$a的值也同时改变了$b的值,所以在执行了

$a = &test();
$a = 5;
登入後複製

以后,$b的值变为了5。

这里是为了让大家理解函数的引用返回才使用静态变量的,其实函数的引用返回多用在对象中。

另附一个PHP官方例子:

<?php
// This is the way how we use pointer to access variable inside the class.
class talker{

	private $data = 'Hi';

	public function & get() {
		return $this->data;
	}

	public function out() {
		echo $this->data;
	}
}

$aa = new talker();
$d = &$aa->get();

$aa->out();
$d = 'How';
$aa->out();
$d = 'Are';
$aa->out();
$d = 'You';
$aa->out();

// the output is "HiHowAreYou"
?>
登入後複製

4.对象的引用

<?php
class a {
	var $abc = "ABC";
}
$b = new a;
$c = $b;
echo $b->abc; // 这里输出ABC
echo $c->abc; // 这里输出ABC
$b->abc="DEF";
echo $c->abc; // 这里输出DEF
?>
登入後複製

以上代码是在PHP5中的运行效果。

在PHP5中对象的赋值是个引用的过程。上列中 $b = new a; $c = $b;  其实等效于 $b = new a; $c = &$b;,PHP5中默认就是通过引用来调用对象, 但有时你可能想建立一个对象的副本,并希望原来的对象的改变不影响到副本 。为了这样的目的,PHP5定义了一个特殊的方法,称为__clone。

自 PHP 5 起,new 自动返回引用,因此在此使用 =& 已经过时了并且会产生 E_STRICT 级别的消息。在PHP4中,对象的赋值是个拷贝过程,如:$b = new a,其中new a产生的是一个匿名的a对象实例,而此时的$b是对这个匿名对象的拷贝。同理 $c = $b,也是对$b内容的一个拷贝。所以在PHP4中,为了节省内存空间,$b = new a 一般会改成引用的模式,即 $b = &new a。

下面再来个 官方 提供的例子:

在PHP5中,你不需要额外添加什么东西就可到达“对象引用”的功能:

<?php
class foo {
	protected $name;

	function __construct($str) {
		$this->name = $str;
	}

	function __toString() {
		return 'my name is "' . $this->name . '" and I live in "' . __CLASS__ . '".' . " ";
	}

	function setName($str) {
		$this->name = $str;
	}
}

class MasterOne {
	protected $foo;

	function __construct($f) {
		$this->foo = $f;
	}

	function __toString() {
		return 'Master: ' . __CLASS__ . '  foo: ' . $this->foo . " ";
	}

	function setFooName($str) {
		$this->foo->setName($str);
	}
}

class MasterTwo {
	protected $foo;

	function __construct($f) {
		$this->foo = $f;
	}

	function __toString() {
		return 'Master: ' . __CLASS__ . '  foo: ' . $this->foo . " ";
	}

	function setFooName($str) {
		$this->foo->setName($str);
	}
}

$bar = new foo('bar');

print(" ");
print("Only Created $bar and printing $bar ");
print($bar);

print(" ");
print("Now $baz is referenced to $bar and printing $bar and $baz ");
$baz = &$bar;
print($bar);

print(" ");
print("Now Creating MasterOne and Two and passing $bar to both constructors");
$m1 = new MasterOne($bar);
$m2 = new MasterTwo($bar);
print($m1);
print($m2);

print(" ");
print("Now changing value of $bar and printing $bar and $baz");
$bar->setName('baz');
print($bar);
print($baz);

print(" ");
print("Now printing again MasterOne and Two");
print($m1);
print($m2);

print(" ");
print("Now changing MasterTwo's foo name and printing again MasterOne and Two");
$m2->setFooName('MasterTwo\'s Foo');
print($m1);
print($m2);

print("Also printing $bar and $baz");
print($bar);
print($baz);
?>
登入後複製

输出:

Only Created $bar and printing $bar
my name is "bar" and I live in "foo".Now $baz is referenced to $bar and printing $bar and $baz
my name is "bar" and I live in "foo".Now Creating MasterOne and Two and passing $bar to both constructors
Master: MasterOne  foo: my name is "bar" and I live in "foo".Master: MasterTwo  foo: my name is "bar" and I live in "foo".Now changing value of $bar and printing $bar and $baz
my name is "baz" and I live in "foo".
my name is "baz" and I live in "foo".Now printing again MasterOne and Two
Master: MasterOne  foo: my name is "baz" and I live in "foo".Master: MasterTwo  foo: my name is "baz" and I live in "foo".Now changing MasterTwo's foo name and printing again MasterOne and Two
Master: MasterOne  foo: my name is "MasterTwo's Foo" and I live in "foo".Master: MasterTwo  foo: my name is "MasterTwo's Foo" and I live in "foo".Also printing $bar and $baz
my name is "MasterTwo's Foo" and I live in "foo".
my name is "MasterTwo's Foo" and I live in "foo".

上个例子解析:

$bar = new foo('bar');
$m1 = new MasterOne($bar);
登入後複製

实例对象$m1与$m2中的$bar是对实例$bar的引用,而非拷贝,这是PHP5中,对象引用的特点,也就是说

  1. $m1或$m2内部,任何对$bar的操作都会影响外部对象实例$bar的相关值。
  2. 外部对象实例$bar的改变也会影响$m1和$m2内部的$bar的引用相关值。

在PHP4中,要实现如上述的用一个对象实例去当着另外一个对象的属性时,其等价代码(即引用调用)类似如下:

class foo{
	var $bar;
	function setBar(&$newBar) {
		$this->bar = $newBar;
	}
}
登入後複製

5.引用的作用

如果程序比较大,引用同一个对象的变量比较多,并且希望用完该对象后手工清除它,个人建议用 "&" 方式,然后用 $var = null 的方式清除。 其它时候还是用PHP5的默认方式吧。另外,PHP5中对于大数组的传递,建议用 "&" 方式,毕竟节省内存空间使用。

6.取消引用

当你 unset 一个引用,只是断开了变量名和变量内容之间的绑定。这并不意味着变量内容被销毁了。例如:

<?php
$a = 1;
$b = &$a;
unset ($a);
?>
登入後複製

不会 unset $b,只是 $a。

7.global 引用

当用 global $var 声明一个变量时实际上建立了一个到全局变量的引用。也就是说和这样做是相同的:

<?php
$var = &$GLOBALS["var"];
?>
登入後複製

这意味着,例如,unset $var 不会 unset 全局变量。

如果在一个函数内部给一个声明为 global 的变量赋于一个引用,该引用只在函数内部可见。可以通过使用 $GLOBALS 数组避免这一点。

Example  在函数内引用全局变量

<?php
$var1 = "Example variable";
$var2 = "";

function global_references($use_globals) {
	global $var1, $var2;
	if (!$use_globals) {
		$var2 = &$var1; // visible only inside the function
	} else {
		$GLOBALS["var2"] = &$var1; // visible also in global context
	}
}

global_references(false);
echo "var2 is set to '$var2'
"; // var2 is set to ''
global_references(true);
echo "var2 is set to '$var2'
"; // var2 is set to 'Example variable'
?>
登入後複製

global $var; 当成是 $var = &$GLOBALS['var']; 的简写。从而将其它引用赋给 $var 只改变了本地变量的引用。

8.$this

在一个对象的方法中,$this 永远是调用它的对象的引用。

//下面再来个小插曲

PHP中对于地址的指向(类似指针)功能不是由用户自己来实现的,是由Zend核心实现的,PHP中引用采用的是“写时拷贝”的原理,就是除非发生写操作,指向同一个地址的变量或者对象是不会被拷贝的。

通俗的讲

1:如果有下面的代码

<?
$a = "ABC";
$b = &$a;
?>
登入後複製

其实此时 $a与$b都是指向同一内存地址 而并不是$a与$b占用不同的内存

2:如果在上面的代码基础上再加上如下代码

<?php
$a = "EFG";
?>
登入後複製

由于$a与$b所指向的内存的数据要重新写一次了,此时Zend核心会自动判断,自动为$b生产一个$a的数据拷贝,重新申请一块内存进行存储PHP的引用(就是在变量或者函数、对象等前面加上&符号)是个高级话题,新手多注意,正确的理解PHP的引用很重要,对性能有较大影响,而且理解错误可能导致程序错误!

很多人误解PHP中的引用跟C当中的指针一样,事实上并非如此,而且很大差别。C语言中的指针除了在数组传递过程中不用显式申明外,其他都需要使用*进行定 义,而PHP中对于地址的指向(类似指针)功能不是由用户自己来实现的,是由Zend核心实现的,PHP中引用采用的是“写时拷贝”的原理,就是除非发生写操作,指向同一个地址的变量或者对象是不会被拷贝的,比如下面的代码:

$a = array('a','c'...'n');
$b = $a;
登入後複製

如果程序仅执行到这里,$a和$b是相同的,但是并没有像C那样,$a和$b占用不同的内存空间,而是指向了同一块内存,这就是PHP和C的差别,并不需要写成$b=&$a才表示$b指向$a的内存,zend就已经帮你实现了引用,并且zend会非常智能的帮你去判断什么时候该这样处理,什么时候不该这样处理。

如果在后面继续写如下代码,增加一个函数,通过引用的方式传递参数,并打印输出数组大小。

function printArray(&$arr) { //引用传递
	print(count($arr));
}
printArray($a);
登入後複製

上面的代码中,我们通过引用把$a数组传入printArray()函数,zend引擎会认为 printArray() 可能会导致对$a的改变,此时就会自动为$b生产一个$a的数据拷贝,重新申请一块内存进行存储。这就是前面提到的“写时拷贝”概念。

如果我们把上面的代码改成下面这样:

function printArray($arr) { // 值传递
	print(count($arr));
}
printArray($a);
登入後複製

上面的代码直接传递$a值到printArray()中,此时并不存在引用传递,所以没有出现写时拷贝。

大家可以测试一下上面两行代码的执行效率,比如外面加入一个循环1000次,看看运行的耗时,结果会让你知道不正确使用引用会导致性能下降30%以上。

自我理解:按传值的话是与函数内的参数无关,相当于局部变量的作用,而按传址(引用)的话却与函数内的参数有关,相当于全局变量的作用。而从性能方面来说,看上面分析就够。

您可能感兴趣的文章

  • php安全编程注意事项
  • 网站空间安全注意事项
  • 在php中分别使用curl的post提交数据的方法和get获取网页数据的方法总结
  • php递归函数中使用return需注意
  • php利用array_flip实现数组键值交换去除数组重复值
  • php利用正则过滤各种标签,空格,换行符的代码
  • ThinkPHP内置模板引擎的使用方法总结
  • PHP 利用 Curl Functions 实现多线程抓取网页和下载文件



本網站聲明
本文內容由網友自願投稿,版權歸原作者所有。本站不承擔相應的法律責任。如發現涉嫌抄襲或侵權的內容,請聯絡admin@php.cn

熱AI工具

Undresser.AI Undress

Undresser.AI Undress

人工智慧驅動的應用程序,用於創建逼真的裸體照片

AI Clothes Remover

AI Clothes Remover

用於從照片中去除衣服的線上人工智慧工具。

Undress AI Tool

Undress AI Tool

免費脫衣圖片

Clothoff.io

Clothoff.io

AI脫衣器

Video Face Swap

Video Face Swap

使用我們完全免費的人工智慧換臉工具,輕鬆在任何影片中換臉!

熱門文章

<🎜>:泡泡膠模擬器無窮大 - 如何獲取和使用皇家鑰匙
3 週前 By 尊渡假赌尊渡假赌尊渡假赌
北端:融合系統,解釋
3 週前 By 尊渡假赌尊渡假赌尊渡假赌
Mandragora:巫婆樹的耳語 - 如何解鎖抓鉤
3 週前 By 尊渡假赌尊渡假赌尊渡假赌

熱工具

記事本++7.3.1

記事本++7.3.1

好用且免費的程式碼編輯器

SublimeText3漢化版

SublimeText3漢化版

中文版,非常好用

禪工作室 13.0.1

禪工作室 13.0.1

強大的PHP整合開發環境

Dreamweaver CS6

Dreamweaver CS6

視覺化網頁開發工具

SublimeText3 Mac版

SublimeText3 Mac版

神級程式碼編輯軟體(SublimeText3)

熱門話題

Java教學
1666
14
CakePHP 教程
1426
52
Laravel 教程
1328
25
PHP教程
1273
29
C# 教程
1253
24
Win11 Xbox Game Bar怎麼會徹底卸載?分享Xbox Game Bar卸載方法 Win11 Xbox Game Bar怎麼會徹底卸載?分享Xbox Game Bar卸載方法 Feb 10, 2024 am 09:21 AM

Win11XboxGameBar怎麼徹底卸載? XboxGameBar是系統中自帶的遊戲平台,它提供了用於遊戲錄製、截圖和社交功能的工具,不過很是佔用內存,也不好卸載,一些小伙伴想要將它卸載掉,但是不這道怎麼徹底卸載,下面就來為大家介紹一下。方法一、使用Windows終端機1、按【Win+X】組合鍵,或【右鍵】點選工作列上的【Windows開始功能表】,在開啟的的選單項目中,選擇【終端機管理員】。 2.用戶帳戶控制窗口,你要允許此應用程式對你的設備進行更改嗎?點選【是】。 3、執行以下指令:Get-AppxP

function是什麼意思 function是什麼意思 Aug 04, 2023 am 10:33 AM

function是函數的意思,是一段具有特定功能的可重複使用的程式碼區塊,是程式的基本組成單元之一,可以接受輸入參數,執行特定的操作,並傳回結果,其目的是封裝一段可重複使用的程式碼,提高程式碼的可重複使用性和可維護性。

「Touch ID 登入」卡在 Mac Touch Bar 上?這是修復 「Touch ID 登入」卡在 Mac Touch Bar 上?這是修復 Apr 15, 2023 pm 12:04 PM

使用Safari圖示修復MacTouchBar上的「TouchID登入」問題強制觸控列重新啟動將解決此問題:在Mac上開啟ActivityMonitor,它位於/Applications/Utilities資料夾中,或者您可以使用Spotlight透過按Command+空白鍵並鍵入ActivityMonitor並返回來啟動它使用活動監視器的搜尋功能並蒐索“觸摸”選擇“TouchBarServer”,然後單擊活動監視器工具列中的(X)退出按鈕選擇“強制退出”以強制

'enumerate()'函數在Python中的用途是什麼? 'enumerate()'函數在Python中的用途是什麼? Sep 01, 2023 am 11:29 AM

在本文中,我們將了解enumerate()函數以及Python中「enumerate()」函數的用途。什麼是enumerate()函數? Python的enumerate()函數接受資料集合作為參數並傳回一個枚舉物件。枚舉物件以鍵值對的形式傳回。 key是每個item對應的索引,value是items。語法enumerate(iterable,start)參數iterable-傳入的資料集合可以作為枚舉物件傳回,稱為iterablestart-顧名思義,枚舉物件的起始索引由start定義。如果我們忽

MySQL.proc表的作用與功能詳解 MySQL.proc表的作用與功能詳解 Mar 16, 2024 am 09:03 AM

MySQL.proc表的功能與功能詳解MySQL是一種流行的關係型資料庫管理系統,開發者在使用MySQL時常常會涉及到預存程序(StoredProcedure)的建立與管理。而MySQL.proc表則是一個非常重要的系統表,它儲存了資料庫中所有的預存程序的相關信息,包括預存程序的名稱、定義、參數等。在本文中,我們將詳細解釋MySQL.proc表的作用與功能

Vue.use函數的用法與作用 Vue.use函數的用法與作用 Jul 24, 2023 pm 06:09 PM

Vue.use函數的用法和作用Vue是一款流行的前端框架,它提供了許多有用的功能和功能。其中之一就是Vue.use函數,它可以讓我們在Vue應用中使用插件。本文將介紹Vue.use函數的用法和作用,並且提供一些程式碼範例。 Vue.use函數的基本用法非常簡單,只需在Vue實例化之前呼叫它,並傳入要使用的插件作為參數。以下是一個簡單的範例://引入並使用插件

在PHP中的clearstatcache()函數 在PHP中的clearstatcache()函數 Sep 07, 2023 am 09:33 AM

clearstatcache()函數用於清除檔案狀態快取。 PHP快取以下函數傳回的資訊−stat()lstat()file_exists()is_writable()is_readable()is_executable()is_file()is_dir()filegroup()fileowner()filesize()filetype()fileperms()這樣做是為了提供更好的性能。語法voidclearstatecache()參數NA傳回值clearstatcache(

在PHP中的file_exists()函數 在PHP中的file_exists()函數 Sep 14, 2023 am 08:29 AM

file_exists方法檢查檔案或目錄是否存在。它接受要檢查的檔案或目錄的路徑作為參數。以下是它的用途-當您需要在處理之前知道文件是否存在時,它非常有用。這樣,在建立新檔案時使用此函數即可知道該檔案是否已存在。語法file_exists($file_path)參數file_path-設定要檢查是否存在的檔案或目錄的路徑。必需。返回file_exists()方法返回。如果檔案或目錄存在,則傳回TrueFalse,如果檔案或目錄不存在範例讓我們看一個檢查「candidate.txt」檔案和即使文件

See all articles