zoukankan      html  css  js  c++  java
  • PHP的重载-使用魔术方法实现

    摘录PHP官网对PHP重载的解释:


    PHP所提供的"重载"(overloading)是指动态地"创建"类属性和方法。我们是通过魔术方法(magic methods)来实现的。
    当调用当前环境下未定义或不可见的类属性或方法时,重载方法会被调用。本节后面将使用"不可访问属性(inaccessible properties)"和"不可访问方法(inaccessible methods)"来称呼这些未定义或不可见的类属性或方法。
    所有的重载方法都必须被声明为 public。

    Note:
    这些魔术方法的参数都不能通过引用传递。

    Note:
    PHP中的"重载"与其它绝大多数面向对象语言不同。传统的"重载"是用于提供多个同名的类方法,但各方法的参数类型和个数不同。

    属性重载
    public __set ( string $name , mixed $value ) : void
    public __get ( string $name ) : mixed
    public __isset ( string $name ) : bool
    public __unset ( string $name ) : void

    在给不可访问属性赋值时,__set() 会被调用。
    读取不可访问属性的值时,__get() 会被调用。
    当对不可访问属性调用 isset() 或 empty() 时,__isset() 会被调用。
    当对不可访问属性调用 unset() 时,__unset() 会被调用。
    参数 $name 是指要操作的变量名称。__set() 方法的 $value 参数指定了 $name 变量的值。
    属性重载只能在对象中进行。在静态方法中,这些魔术方法将不会被调用。所以这些方法都不能被 声明为 static。

    Note:
    因为 PHP 处理赋值运算的方式,__set() 的返回值将被忽略。类似的, 在下面这样的链式赋值中,__get() 不会被调用:
    $a = $obj->b = 8;

    Note:
    在除 isset() 外的其它语言结构中无法使用重载的属性,这意味着当对一个重载的属性使用 empty() 时,重载魔术方法将不会被调用。
    为避开此限制,必须将重载属性赋值到本地变量再使用 empty()。

    Example #1 使用 __get(),__set(),__isset() 和 __unset() 进行属性重载

    class PropertyTest
    {
        /**  被重载的数据保存在此  */
        private $data = array();
    
        /**  重载不能被用在已经定义的属性  */
        public $declared = 1;
    
        /**  只有从类外部访问这个属性时,重载才会发生 */
        private $hidden = 2;
    
        public function __set ($name, $value)
        {
            $this->data[$name] = $value;
        }
    
        public function __get ($name)
        {
            if (isset($this->$name)) {
                return $this->$name;
            }
            if (array_key_exists($name, $this->data)) {
                return $this->data[$name];
            }
            //产生一条回溯跟踪
            $trace = debug_backtrace();
            //抛出异常
            trigger_error('Undefined property via __get(): ' . $name . ' in ' . $trace[0]['file'] . ' on line ' . $trace[0]['line'], E_USER_NOTICE);
            return null;
        }
    
        public function __isset ($name)
        {
            return isset($this->data[$name]);
        }
    
        public function __unset ($name)
        {
            unset($this->data[$name]);
        }
    
        /**  非魔术方法  */
        public function getHidden ()
        {
            return $this->hidden;
        }
    }
    
    $obj = new PropertyTest;
    
    //输出不存在的a变量,走到__get()中,会抛出异常
    echo $obj->a;
    
    //对不存在的a变量赋值为1,会走到__set()中
    $obj->a = 1;
    
    //再次输出a变量,由于上面已经对其__set(),所以这是可以访问到a的值为1
    echo $obj->a;
    
    //此时使用isset()对不存在的a变量进行运算时,会走到__isset()中,由于上面已经对其__set(),所以是true
    var_dump(isset($obj->a));
    
    //对a进行unset()时,会走到__unset()中
    unset($obj->a);
    
    //再对其进行isset(),此时已经不存在了
    var_dump(isset($obj->a));
    
    //访问private 属性的变量,会进入__get()中
    echo $obj->hidden;


    方法重载
    public __call ( string $name , array $arguments ) : mixed
    public static __callStatic ( string $name , array $arguments ) : mixed

    在对象中调用一个不可访问方法时,__call() 会被调用。
    在静态上下文中调用一个不可访问方法时,__callStatic() 会被调用。
    $name 参数是要调用的方法名称。
    $arguments 参数是一个枚举数组,包含着要传递给方法 $name 的参数。

    Example #2 使用 __call() 和 __callStatic() 对方法重载

    class MethodTest
    {
        /**
         * 调用不存在的方法时进入此处
         * @param $name
         * @param $arguments
         */
        public function __call ($name, $arguments)
        {
            // 注意: $name 的值区分大小写
            $info = [
                'name' => $name,
                'arguments' => $arguments,
            ];
            print_r($info);
        }
    
        /**
         * PHP 5.3.0之后版本
         * 调用不存在的静态方法时,进入此处
         */
        public static function __callStatic ($name, $arguments)
        {
            // 注意: $name 的值区分大小写
            $info = [
                'name' => $name,
                'arguments' => $arguments,
            ];
            print_r($info);
        }
    }
    
    $arguments = ['A', 'B', 'C'];
    
    $obj = new MethodTest;
    $obj->test(...$arguments);
    
    MethodTest::test(...$arguments);  // PHP 5.3.0之后版本
    
    /*
     *  以上两个都输出:
     *  Array
        (
            [name] => test
            [arguments] => Array
                (
                    [0] => A
                    [1] => B
                    [2] => C
                )
    
        )
     */



  • 相关阅读:
    微服务架构总结
    微服务-网关服务
    HttpClient-RestTemplate-Feign
    RPC和REST
    Springmvc的拦截器执行顺序及各方法作用
    秒杀系统优化方案(下)吐血整理
    秒杀系统优化方案(上)吐血整理
    分布式session的管理
    缓存设计——缓存和数据库的数据一致性
    个人理解的javascript作用域链与闭包
  • 原文地址:https://www.cnblogs.com/deverz/p/11093218.html
Copyright © 2011-2022 走看看