///////////////////////////////////////////////////// //@date: 2013-02-21 //引用定位 //许多PHP的语法结构是通过引用机制实现的,所以上述有关引用绑定的事切都适用于这些结构,一些结构,例如引用传递和返回, //global 引用 //当用global $var 声明一个变量时实现上建立了一个全局变量的引用,也就是说和这样做是相同的 //$var = & $GLOBALS['var']; //$this 在一个对象的方法中,$this永远是调用它的对象的引用 // $a = 1; // $b =& $a; // $c =2; // $d = 3; // $e = array($a); // function is_reference($var){ // $val = $GLOBALS[$var]; // $tmpArray = array(); // /** // * Add keys/values without reference // */ // foreach($GLOBALS as $k => $v){ // if(!is_array($v)){ // $tmpArray[$k] = $v; // } // } // /** // * Change value of rest variables // */ // foreach($GLOBALS as $k => $v){ // if($k != 'GLOBALS' && $k != '_POST' && $k != '_GET' && $k != '_COOKIE' // && $k != '_FILES') // { // $is_a = !is_array($v); // if($k != $var && !is_array($v)){ // echo "var:{$var}===={$k} is_a:{$is_a}<BR><BR>"; // echo "usleep"; // usleep(1); // $GLOBALS[$k] = md5(microtime()); // echo $GLOBALS[$k]."<BR><BR>"; // } // } // } // echo $val."<BR><BR>"; //1 // echo $GLOBALS[$var]."<BR><BR>"; // $bool = $val != $GLOBALS[$var]; // /** // * Restore defaults values // */ // foreach($tmpArray as $k => $v){ // $GLOBALS[$k] = $v; // } // return $bool; // } // var_dump(is_reference('a')); // var_dump(is_reference('b')); // var_dump(is_reference('c')); // var_dump(is_reference('d')); // // 基本理解,明天晚上重新一次 // $a = 1; // $b = &$a; // $c = 2; // $d = 3; // $e = array($a); // function is_reference($var) // { // $val = $GLOBALS[$var]; // $tmpArray = array(); // foreach ($GLOBALS as $k => $v) { // if(!is_array($v)){ // $tmpArray[$k] = $v; // } // } // foreach($GLOBALS as $k=>$v) // { // if($k != "GLOBALS" && $k !="_POST" && $k !="_GET" // && $k != "_COOKIE" && $k !="_FILES") // { // if($k != $var && !is_array($v)) // { // usleep(1); // $GLOBALS[$k] = md5(microtime()); // } // } // } // //echo "\$var:{$var} ".$GLOBALS[$var]; // $bool = $val != $GLOBALS[$var]; // foreach ($tmpArray as $key => $value) { // $GLOBALS[$key] = $value; // } // return $bool; // } // var_dump(is_reference('a')); // echo "<BR><BR>"; // var_dump(is_reference('b')); // echo "<BR><BR>"; // var_dump(is_reference('c')); // echo "<BR><BR>"; // var_dump(is_reference('d')); // echo "<BR><BR>"; //如果你要检查,如果两个变量引用对方(即指向相同的内存),你可以使用这样的功能: // function same_type(&$var1, &$var2) // { // return gettype($var1) === gettype($var2); // } // function is_ref(&$var1, &$var2) // { // if(!same_type($var1, $var2)) // return false; // $same = false; // //数组 // if(is_array($var1)) // { // //获取或生成一个唯一的key值 // do{ // $key = uniqid("is_ref_", true); // }while(array_key_exists($key, $var1)); // if(array_key_exists($key, $var2)){ // return false; // } // $data = uniqid("is_ref_data_", true); // $var1[$key] = & $data; //设置$var1[$key]的值 // //这里也就是如果$var1与$var2是引用关系 // //也就是关联到同一个内存地址 // if(array_key_exists($key, $var2)){ // if($var2[$key] === $data){ // $same = true; // } // } // unset($var1[$key]); // //如果是对象 // }elseif(is_object($var1)) // { // //如果不是同一个类型,直接返回 // //get_class返类名 // if(get_class($var1) !== get_class($var2)) // return false; // //get_object_vars返回对象的所有属性 // $obj1 = array_keys(get_object_vars($var1)); // $obj2 = array_keys(get_object_vars($var2)); // do{ // $key = uniqid("is_ref_",true); // }while(in_array($key, $obj1)); // $data = uniqid("is_ref_", true); // $obj1->$key = &$data; // if(isset($var2->$key)) // { // if($var2->$key === $data) // $same = true; // } // unset($var1->$key); // //判断是否为资源 // }elseif(is_resource($var1)) // { // if(get_resource_type($var1) !== get_resource_type($var2)) // return false; // return ((string)$var1) === ((string)$var2); // //其它情况,也就是基本数据类型的情况 // }else{ // if($var1 !== $var2) // return false; // do{ // $key = uniqid("is_ref_", true); // }while($key === $var1); // $tmp = $var1; // $var1 = $key; // $same = $var1 === $var2; // $var1 = $tmp; // } // return $same; // } // $a = 5; // $b = 5; // var_dump(is_ref($a, $b)); //false // echo "<BR>======================<BR><BR>"; // $a = 5; // $b = $a; //copy // var_dump(is_ref($a, $b)); //false // echo "<BR>======================<BR><BR>"; // $a = 5; // $b = &$a; // var_dump(is_ref($a, $b)); //true // echo "<BR>======================<BR><BR>"; // $a = array(); // var_dump(is_ref($a, $a)); //true // echo "<BR>======================<BR><BR>"; // $a[] = &$a; // var_dump(is_ref($a, $a[0])); // echo "<BR>======================<BR><BR>"; // $b = array(array()); // var_dump(is_ref($b,$b)); //true // var_dump(is_ref($b, $b[0])); //false; // echo "<BR>======================<BR><BR>"; // $b = array(); // $b[] = $b; // var_dump(is_ref($b, $b)); //true // var_dump(is_ref($b, $b[0])); // var_dump(is_ref($b[0], $b[0][0])); // echo "<BR>======================<BR><BR>"; // var_dump($a); // var_dump($b); // //*请注意PHP的内部行为,似乎做之前,实际上复制的变量的引用赋值!因此,你会得到一个数组,其中包含一个(不同的)递归阵列的最后一个测试用例,而不是一个数组,包含一个空数组,你可以期望。 // class Test // { // var $monkeys = 0; // function doFoodbar() // { // $var = 'this'; // $$var->monkeys ++; //这样操作是失败的 // //在这条线将失败 // } // function addMonkeys() // { // $this->monkeys++; // } // } // $obj = new Test; // $obj->doFoodbar(); // var_dump($obj->monkeys); // $obj->addMonkeys(); // echo "<BR><BR>"; // var_dump($obj->monkeys); // class A{}; // $oA1 = new A(); // $roA = & $oA1; // if($roA == $oA1){ // echo "same"; // }else // echo "not same"; // echo "<BR><BR>"; // $oA2 = new A(); // $roA = & $roA2; //$roA2这个对像为空呢 // if($roA == $oA1) // echo "same"; // else // echo "not same"; // echo "<BR><BR>"; // class y{ // public $d; // } // $A = new y; // $A->d = 18; // var_dump($A);echo "<BR><BR>"; // $B = $A; // $C = &$A; // $B->d = 1234; // //这不是一个明确的(=&)引用赋值, // //但是,$ A$ B指的同一个实例 // //虽然他们是不等价的名 // var_dump($B);echo "<BR><BR>"; //让我不能理解的怎么B也相当于引用了 // var_dump($C);echo "<BR><BR>"; // var_dump($A);echo "<BR><BR>"; // $A = new y; // $A->d = 250; // var_dump($B);echo "<BR><BR>"; //1234 // var_dump($C);echo "<BR><BR>"; //1234这样理解是错的,因为在上一步中 // //$C = &$A;所以这里的$C也是250 // var_dump($A);echo "<BR><BR>"; //250 // class yy // { // public $d; // function yy($d) // { // $this->d = $x; // } // } // function modify($v) // { // $v->d = 1225; // } // $A = new yy(3); // var_dump($A); //3; // modify($A); // var_dump($A); //3 //虽然在一般情况下,声明一个正式的参数$V在上面所示的“修改”的功能,意味着 //实际的参数$ A时,通过调用的功能,不被修改,这是不当$ A是一个对象实例。 class myClass { public $datum; public $other; function &MyRef($d) { $this->datum = $d; return $this; //返回一个引用对象 } } $a = new myClass(); $b = $a->MyRef(25); print_r($a);echo "<BR><BR>"; //25 print_r($b);echo "<BR><BR>"; //25 $b->other =50; print_r($a);echo "<BR><BR>"; //50 print_r($b);echo "<BR><BR>"; //50