当前位置: 首页 > news >正文

php 各种魔术函数的触发条件

2024.3.20

1、__construct()

__construct() 用于在创建对象时自动触发

当使用 new 关键字实例化一个类时,会自动调用该类的 __construct() 方法

<?php
class MyClass {public function __construct() {echo "已触发 __construct 一次";}
}$obj = new MyClass();  // 创建对象时会输出 "已触发 __construct 一次"
?>

2、__destruct()

__destruct() 用于在对象被销毁时自动触发

对象的销毁对象的引用计数减少为零来触发

<?php
class MyClass {public function __construct() {echo "已触发 __construct 一次\n";}public function __destruct() {echo "已触发 __destruct 一次\n";}
}$obj = new MyClass();    //会触发 __construct
$a = unserialize($obj) //会触发 __destruct
?>
//由于php存在自动回收机制,在代码结束后会销毁 $obj 从而再一次触发 __destruct
//所以这段代码最终互返回两次"已触发 __destruct 一次"

3、__sleep()

序列化serialize() 函数会检查类中是否存在一个魔术方法sleep()。如果存在,该方法会先被调用,然后才执行序列化操作。

此功能可以用于清理对象,并返回一个包含对象中所有应被序列化的变量名称的数组

如果该方法未返回任何内容,则 NULL 被序列化,并产生一个 E_NOTICE 级别的错误

<?php
class test
{public $var_1;public $var_2;public function __sleep()   //在对象被序列化时触发{echo "已触发 __sleep() 一次\n";return ['var_1','var_2'];}
}$a = new test();
$a -> var_1 = 'var1';
$a -> var_2 = 'var2';
$b = serialize($a);     //会触发 __sleep
var_dump($b);
?>

4、__weekup()

__weekup() 用于在反序列化对象时自动调用

 unserialize() 会检查是否存在一个 wakeup() 方法,

如果存在,则会先调用wakeup()方法,预先准备对象需要的资源,返回void

常用于反序列化操作中重新建立数据库连接或执行其他初始化操作

<?php
class test
{public $var_1;public $var_2;public $var_wakeup;public function __wakeup()   //在对象被反序列化时触发{echo "已触发 __wakeup 一次\n";$this -> var_wakeup = $this -> var_1;}
}$a = new test();
$a -> var_1 = 'var1';
$a -> var_2 = 'var2';
$b = serialize($a);
$c = unserialize($b);   //会触发 __wakeup
var_dump($c);
?>

5、__tostring()

__tostring() 在对象被当做字符串处理时自动调用

比如echo、==、preg_match()

<?php
class test
{public $var_1;public function __tostring()   //对象被当做字符串处理时触发{echo "已触发 __tostring 一次\n";return '1';}
}$a = new test;
$a == '123'; //会触发 __wakeup
echo $a; //会触发 __wakeup
?>

6、__invoke()

__invoke() 在对象被当做函数处理时自动调用

<?php
class test
{public $var_1='1';public function __invoke()   //对象被当做函数处理时触发{echo "已触发 __invock 一次\n";}
}$a = new test('1');
$a(); //以函数的形式处理 $a 从而触发__invoke
?>

7、__call()

__call() 在调用一个不存在的方法时触发

<?php
class test
{public function __call($method, $args)   //当调用一个不存在的方法时触发{echo "已触发 __call 一次\n";}
}$a = new test;
$a -> nothingness('');    //因为nothingness()方法不存在,所以触发了 __call
?>

8、__callStatic()

 __callStatic() 在静态调用或调用成员常量时使用的方法不存在时触发

<?php
class test
{static public function __callStatic($method, $args){echo "已触发 __callStatic 一次\n"; }
}$a = new test(1);
$a::nothingness('a');
?>

9、__set()

__set() 在给不存在的成员属性赋值时触发

<?php
class test
{public $var_1;public $temp;public function __construct($var)   //在对象被创建时触发{echo "已触发 __construct 一次\n";$this -> var_1 = $var;$this -> var_nothingness = '114514';}public function __set($name, $value)    //在给不存在的成员属性赋值时触发{echo "已触发 __set 一次\n";}
}$a = new test(1);
?>

10、__isset()

__isset() 在对不可访问属性使用 isset() 或empty() 时会被触发

<?php
class temp
{private $no;public function __isset($name)    //在给不存在的成员属性赋值时触发{echo "已触发 __isset 一次\n";}
}
class test
{public $var;public $temp;public function __construct($var){echo "已触发 __construct 一次\n";$this -> var_1 = $var;$temp = new temp;empty($temp->no);    //用于触发 __isset}
}$a = new test(1);
?>

 11、__unset()

__unset() 在对不可访问属性使用 unset() 时会被触发

<?php
class temp
{private $no;public function __unset($name)    //对不可访问属性使用 unset() 会被触发{echo "已触发 __unset 一次\n";}
}
class test
{public $var;public $temp;public function __construct($var){echo "已触发 __construct 一次\n";$this -> var_1 = $var;$temp = new temp;unset($temp->no);}
}$a = new test(1);?>

12、__clone()

__clone() 当使用 clone 关键字拷贝完成一个对象后就会触发 

<?php
class test
{public $var;public function __construct($var){echo "已触发 __construct 一次\n";$this -> var_1 = $var;clone($this);   //用于触发 __clone}public function __clone()    //当使用 clone 关键字拷贝完成一个对象后就会触发{echo "已触发 __clone 一次\n";}
}$a = new test(1);
?>

13、__get()

__get() 当尝试访问不可访问属性时会被自动调用

<?php
class Test
{private $data = array();public function __construct($var){echo "已触发 __construct 一次\n";$this->var_1 = $var;}public function __get($name){echo "已触发 __get 一次\n";if (isset($this->data[$name])) {return $this->data[$name];}}public function __set($name, $value){echo "已触发 __set 一次\n";$this->data[$name] = $value;}
}$a = new Test(1);
$a->v = 'a';  // 设置属性
echo $a->v;   // 访问属性
var_dump($a);
?>

警告:以下代码是个人研究时写的,异常混乱,没有任何的逻辑和结果可言,大佬勿看,看完容易高血压,萌新勿看,看完容易裂开

<?phpclass temp
{private $no;public function __isset($name)    //对不可访问属性使用 isset() 或empty() 会被触发{echo "已触发 __isset 一次\n";}public function __unset($name)    //对不可访问属性使用 unset() 会被触发{echo "已触发 __unset 一次\n";}
}
class test
{public $var_1;public $var_2;public $var_wakeup;public $temp;private $data = array();public function __construct($var)   //在对象被创建时触发{echo "已触发 __construct 一次\n";$this -> var_1 = $var;} public function __destruct()   //在对象被销毁时触发{echo "已触发 __destruct 一次\n";$this();    //用于触发 __invoke}public function __sleep()   //在对象被序列化时触发{echo "已触发 __sleep() 一次\n";$this::nothingness('');   //用于触发 __call$this -> var_nothingness = '114514';    //用于触发 __setreturn ['var_1','var_2'];}public function __wakeup()   //在对象被反序列化时触发{echo "已触发 __wakeup 一次\n";if($this -> var_2 == '') //用于触发 __tostring{echo "比较通过\n";}$this -> var_wakeup = $this -> var_1;}public function __tostring()   //对象被当做字符串处理时触发{echo "已触发 __tostring 一次\n";return '1';}public function __invoke()   //对象被当做函数处理时触发{echo "已触发 __invock 一次\n";}public function __call($method, $args)   //当调用一个不存在的方法时触发{echo "已触发 __call 一次\n";clone($this);   //用于触发 __clone}static public function __callStatic($method, $args)   //在静态调用或调用成员常量时使用的方法不存在时触发{echo "已触发 __callStatic 一次\n";$temp = new temp;isset($temp -> no);     //用于触发 __issetunset($temp -> no);     //用于触发 __unset}public function __clone()    //当使用 clone 关键字拷贝完成一个对象后就会触发{echo "已触发 __clone 一次\n";}public function __get($name) //当尝试访问不可访问属性时会被自动调用{echo "已触发 __get 一次\n";$this-> name = "$name";     //用于触发 __set}public function __set($name, $value)    //在给不存在的成员属性赋值时触发{echo "已触发 __set 一次\n";$this->data[$name] = $value;}
}//会触发一次 __construct 并在对象被销毁时触发一次 __destruct
//然后 __destruct 中的 $this() 又会触发 __invoke
$a = new test('var1');//会触发一次 __construct 并在对象被销毁时触发一次 __destruct
//然后 __destruct 中的 $this() 又会触发一次 __invoke
$a -> var_2 = new test('var2');//因为 $a 是一个对象,$a -> var_2 又是一个对象,会触发两次 __sleep
//__sleep中会调用不存在的函数和给不存在的值赋值,又会触发 __call 和 __set
//__call又会触发一次 __clone 和一次 __destruct
$b = serialize($a);//同理会触发会先触发两次 __wakeup 再触发两次 __destruct
//然后 __destruct 中的 $this() 又会触发 __invoke
//__wakeup中有比较,所以又会触发一次 __tostring
//注意:__tostring 是由 $a -> var_2 触发的,而 $a -> var_2 -> var_2 由于值为 NULL 是不会触发__tostring的
$c = unserialize($b);//会触发一次 __callStatic
//__callStatic 里面的 isset 会触发一次 __isset,unset 会触发一次 __unset
$c::nothingness('');//会触发一次 __get
//__get 又触发一次 __set
echo $c -> v;//var_dump($c);

http://www.lryc.cn/news/324139.html

相关文章:

  • Linux的学习之路:2、基础指令(1)
  • 0103设计算法-算法基础-算法导论第三版
  • [NCTF2019]SQLi ---不会编程的崽
  • 上位机开发 halcon坐标转轴坐标
  • [数据结构]二叉树(下)
  • 动手学深度学习|notebook教程
  • C#面:简述 .NET Framework 类库中的“命名空间”
  • android.os.TransactionTooLargeException解决方案,Kotlin
  • ChatGPT智能聊天系统源码v2.7.6全开源Vue前后端+后端PHP
  • 汇丰:当前的美股是泡沫吗?
  • 颠覆传统:Web3如何塑造未来的数字经济
  • iOS模拟器 Unable to boot the Simulator —— Ficow笔记
  • 使用 Flink + Faker Connector 生成测试数据压测 MySQL
  • Android单片机硬件通信《GPIO通信》
  • C# WPF编程-事件
  • C语言 预处理器 注释 基本案例讲解
  • Flutter学习10 - Json解析与Model使用
  • Clickhouse异常:Exception: No operation equals between Decimal(X, X) and Float64
  • 会员中心微服务
  • element el-dialog里再调用其他组件,查找不到组件的方法
  • 【深度学习】四种天气分类 模版函数 从0到1手敲版本
  • Linux文件 profile、bashrc、bash_profile区别
  • blender记一下法线烘焙
  • 【LabVIEW FPGA入门】FPGA 存储器(Memory)
  • vue3+element Plus form 作为子组件,从父组件如何赋值?
  • Kafka系列之:Exactly-once support
  • Spring Boot2
  • 【idea做lua编辑器】IDEA下lua插件报错编辑器打不开(同时安装EmmyLua和Luanalysis这2个插件就报错,保留EmmyLua插件即可)
  • SpringCloud之网关组件Gateway学习
  • 全球大型语言模型(LLMS)现状与比较