Rumah > pembangunan bahagian belakang > masalah PHP > Apakah kaedah ajaib php

Apakah kaedah ajaib php

青灯夜游
Lepaskan: 2023-03-17 17:32:01
asal
4272 orang telah melayarinya

Kaedah terbina dalam PHP yang bermula dengan garis bawah berganda "__" dipanggil kaedah ajaib, termasuk "__construct()", "__set()", "__get()", "__isset() ", "__unset()", "__sleep()", "__wakeup()", "__call()", "__invoke()", dll.; antaranya "__construct()" ialah kaedah pembinaan kelas dan ialah langkah pertama selepas objek dicipta Kaedah yang dipanggil secara automatik oleh objek.

Apakah kaedah ajaib php

Persekitaran pengendalian tutorial ini: sistem windows7, versi PHP8, komputer DELL G3

Kaedah ajaib dalam php

Dalam PHP, kaedah yang dinamakan bermula dengan garis bawah berganda (__) dipanggil kaedah ajaib dalam PHP, dan ia memainkan peranan yang sangat penting dalam PHP. Kaedah sihir termasuk:

方法名 描述
__construct() 类的构造函数
__destruct() 类的析构函数
__call($funName, $arguments) 当调用一个未定义或不可达方法时, __call() 方法将被调用。
__callStatic($funName, $arguments) 当调用一个未定义或不可达的静态方法时, __callStatic() 方法将被调用。
__get($propertyName) 当获取一个类的成员变量时, __get() 方法将被调用。
__set($property, $value) 当赋值一个类的成员变量时, __set() 方法将被调用。
__isset($content)  当调用 isset() 或 empty() 对一个未定义或不可达的成员赋值时, __isset() 方法将被调用。
__unset($content) 当调用 reset() 对一个未定义或不可达的成员更新时, __unset() 方法将被调用。
__sleep() 当执行序列化 serialize() 时,__sleep() 方法将首先被调用。
__wakeup() 当执行反序列化 deserialization() 时, __wakeup() 方法将首先被调用。
__toString() 当使用 echo 方法直接输出显示对象时,__toString() 方法首先被调用。
__invoke() 使用调用函数(function)访问一个对象时, __invoke() 方法将首先被调用。
__set_state($an_array) 当调用 var_export() 方法时,__set_state() 方法将被调用。
__clone() 当对象被复制赋值时,__clone() 方法将被调用。
__autoload($className) 试图载入一个未定义的类时调用。
__debugInfo() 输出 debug 信息。

Artikel ini akan menggunakan beberapa contoh untuk menunjukkan penggunaan kaedah sihir PHP.

1.__construct()

Apabila objek dicipta, kaedah pembina kelas PHP ialah kaedah pertama yang dipanggil. Setiap kelas mempunyai kaedah pembina. Jika anda tidak mentakrifkannya secara eksplisit dalam kelas, akan ada pembina kelas tanpa hujah lalai, walaupun ia tidak akan ditakrifkan dalam kelas.

1) Aplikasi kaedah pembina

Kaedah pembina kelas biasanya digunakan untuk melaksanakan beberapa tugas pemula, seperti memulakan dan memberikan nilai kepada ahli semasa mencipta objek.

2) Format pengisytiharan kaedah pembina dalam kelas

function __constrct([parameter list]){

    方法具体实现 //通常为成员变量初始赋值。

}
Salin selepas log masuk

Nota: Hanya satu pembina boleh diisytiharkan dalam kebanyakan kelas. Kerana, PHP tidak menyokong beban berlebihan pembina.

Berikut ialah contoh lengkap:

<?php
    class Person
    {                                                                     
            public $name;       
            public $age;       
            public $sex;       

        /**
         * 明确定义含参的构造方法
         */                                                                                       
        public function __construct($name="", $sex="Male", $age=22)
        {     
            $this->name = $name;
            $this->sex = $sex;
            $this->age = $age;
        }

        /**
         * say 方法定义
         */
        public function say()
        {
            echo "Name:" . $this->name . ",Sex:" . $this->sex . ",Age:" . $this->age;
        }   

    }
Salin selepas log masuk

Buat objek $Person1 tanpa parameter.

$Person1 = new Person();
echo $Person1->say(); //显示:Name:,Sex:Male,Age:22
Salin selepas log masuk

Buat objek $Person2 dengan satu hujah "Jams".

$Person2 = new Person("Jams");
echo $Person2->say(); // 显示: Name: Jams, Sex: Male, Age: 22
Salin selepas log masuk

Panggil dengan 3 parameter untuk mencipta objek $Person3.

$Person3 = new Person ("Jack", "Male", 25);
echo $Person3->say(); // 显示:Name: Jack, Sex: Male, Age: 25
Salin selepas log masuk

__destruct()

Pemusnah ialah lawan kepada pembina.

Pemusnah membenarkan anda melakukan beberapa operasi sebelum memusnahkan objek, seperti menutup fail, mengosongkan set keputusan, dsb.

Destructor ialah ciri baharu yang diperkenalkan dalam PHP 5.

Pemusnah diisytiharkan serupa dengan pembina, bermula dengan dua garis bawah, dan namanya ditetapkan kepada __destruct().

Pengisytiharan pemusnah

function __destruct()
{
    //method body
}
Salin selepas log masuk

Pemusnah tidak boleh mengambil parameter.

Penggunaan pemusnah

Pemusnah biasanya tidak biasa dalam kelas. Ia adalah bahagian pilihan kelas dan biasanya digunakan untuk menyelesaikan beberapa tugas pembersihan sebelum kelas dimusnahkan.

Ini ialah contoh penggunaan pemusnah:

<?php
class Person{     

    public $name;         
    public $age;         
    public $sex;         

    public function __construct($name="", $sex="Male", $age=22)
    {   
        $this->name = $name;
        $this->sex  = $sex;
        $this->age  = $age;
    }

    /**
     * say method
     */
    public function say()
    {
        echo "Name:".$this->name.",Sex:".$this->sex.",Age:".$this->age;
    }   

    /**
     * declare a destructor method
     */
    public function __destruct()
    {
            echo "Well, my name is ".$this->name;
    }
}

$Person = new Person("John");
unset($Person); //destroy the object of $Person created above
Salin selepas log masuk

Hasil keluaran

Well, my name is John
Salin selepas log masuk

__call()

Kaedah ini menerima dua parameter . Parameter pertama ialah nama kaedah yang tidak ditentukan, dan parameter kedua ialah tatasusunan parameter yang dihantar ke kaedah

Apabila menggunakan

function __call(string $function_name, array $arguments)
{
    // method body
}
Salin selepas log masuk

untuk memanggil kaedah yang tidak ditentukan dalam atur cara, __call() kaedah akan dipanggil.

Contoh

<?php
class Person
{                             
    function say()
    {
           echo "Hello, world!<br>";
    }     

    function __call($funName, $arguments)
    {
          echo "The function you called:" . $funName . "(parameter:" ;  // Print the method's name that is not existed.
          print_r($arguments); // Print the parameter list of the method that is not existed.
          echo ")does not exist!!<br>\n";                   
    }                                         
}
$Person = new Person();           
$Person->run("teacher"); // If the method which is not existed is called within the object, then the __call() method will be called automatically.
$Person->eat("John", "apple");             
$Person->say();
Salin selepas log masuk

Paparkan hasil

The function you called: run (parameter: Array([0] => teacher)) does not exist!
The function you called: eat (parameter: Array([0] => John[1] => apple)) does not exist!
Hello world!
Salin selepas log masuk

4 __callStatic()

Apabila kaedah statik yang tidak ditentukan dipanggil dalam atur cara, <.> kaedah akan dipanggil secara automatik. __callStatic()

digunakan sama dengan __callStatic() . Berikut ialah contoh: __call()

<?php
class Person
{
    function say()
    {
        echo "Hello, world!<br>";
    }

    public static function __callStatic($funName, $arguments)
    {
        echo "The static method you called:" . $funName . "(parameter:" ;  // 打印出未定义的方法名。
        print_r($arguments); // 打印出未定义方法的参数列表。
        echo ")does not exist!<br>\n";
    }
}
$Person = new Person();
$Person::run("teacher"); // 如果此项目内不存在的方法被调用了,那么 __callStatic() 方法将被自动调用。
$Person::eat("John", "apple");
$Person->say();
Salin selepas log masuk
Hasil pelaksanaan adalah seperti berikut:

The static method you called: run (parameter: Array([0] => teacher)) does not exist!
The static method you called: eat (parameter: Array([0] => John[1] => apple)) does not exist!
Hello world!
Salin selepas log masuk
__get()

Apabila anda cuba mengakses sifat peribadi objek secara luaran, aplikasi akan membuang pengecualian dan menamatkan larian. Kita boleh menyelesaikan masalah ini menggunakan kaedah

. Kaedah ini boleh mendapatkan nilai harta persendirian dari luar objek. Contohnya, __get

<?php
class Person
{
    private $name;
    private $age;

    function __construct($name="", $age=1)
    {
        $this->name = $name;
        $this->age = $age;
    }

    public function __get($propertyName)
    {   
        if ($propertyName == "age") {
            if ($this->age > 30) {
                return $this->age - 10;
            } else {
                return $this->$propertyName;
            }
        } else {
            return $this->$propertyName;
        }
    }
}
$Person = new Person("John", 60);   // Instantiate the object with the Person class and assign initial values to the properties with the constructor.
echo "Name:" . $Person->name . "<br>";   // When the private property is accessed, the __get() method will be called automatically,so we can get the property value indirectly.
echo "Age:" . $Person->age . "<br>";    // The __get() method is called automatically,and it returns different values according to the object itself.
Salin selepas log masuk
Hasilnya ditunjukkan seperti berikut

Name: John
Age: 50
Salin selepas log masuk
6. digunakan untuk menetapkan sifat peribadi kelas. Selepas sifat yang tidak ditentukan ditetapkan, kaedah

set() dicetuskan dan parameter yang diluluskan ialah nama dan nilai sifat yang ditetapkan.

Berikut ialah kod demo:

Hasil jalankan kod:

<?php
class Person
{
    private $name;
    private $age;

    public function __construct($name="",  $age=25)
    {
        $this->name = $name;
        $this->age  = $age;
    }

    public function __set($property, $value) {
        if ($property=="age")
        {
            if ($value > 150 || $value < 0) {
                return;
            }
        }
        $this->$property = $value;
    }

    public function say(){
        echo "My name is ".$this->name.",I'm ".$this->age." years old";
    }
}

$Person=new Person("John", 25); //请注意,类初始化并为“name”和“age”分配初始值。
$Person->name = "Lili";     // "name" 属性值被成功修改。如果没有__set()方法,程序将报错。
$Person->age = 16; // "age"属性修改成功。
$Person->age = 160; //160是无效值,因此修改失败。
$Person->say();  //输出:My name is Lili, I'm 16 years old。
Salin selepas log masuk
7 __isset()

Menggunakan __isset() Sebelum kaedah, izinkan saya menerangkan penggunaan kaedah isset(). Kaedah isset() digunakan terutamanya untuk menentukan sama ada pembolehubah ini ditetapkan.
My name is Lili, I'm 16 years old
Salin selepas log masuk

Jika kaedah isset() digunakan di luar objek, terdapat dua situasi:

Jika parameter ialah harta awam, anda boleh menggunakan kaedah isset() untuk menentukan sama ada parameter ditetapkan harta.

Kaedah isset() tidak akan berfungsi jika parameter ialah harta peribadi.
  1. Jadi untuk harta persendirian, adakah cara untuk mengetahui sama ada ia ditetapkan? Sudah tentu, selagi kaedah __isset() ditakrifkan dalam kelas, kaedah isset() boleh digunakan di luar kelas untuk menentukan sama ada harta persendirian ditetapkan.
  2. Kaedah __isset() dipanggil apabila isset() atau empty() dipanggil pada harta yang tidak ditentukan atau tidak boleh diakses. Berikut ialah contoh:

Hasil pelaksanaan kod adalah seperti berikut:

<?php
class Person
{
    public $sex;
    private $name;
    private $age;

    public function __construct($name="",  $age=25, $sex=&#39;Male&#39;)
    {
        $this->name = $name;
        $this->age  = $age;
        $this->sex  = $sex;
    }

    /**
     * @param $content
     *
     * @return bool
     */
    public function __isset($content) {
        echo "The {$content} property is private,the __isset() method is called automatically.<br>";
        echo  isset($this->$content);
    }
}

$person = new Person("John", 25); // Initially assigned.
echo isset($person->sex),"<br>";
echo isset($person->name),"<br>";
echo isset($person->age),"<br>";
Salin selepas log masuk
8 __unset()

adalah serupa dengan
1
The name property is private,the __isset() method is called automatically.
1
The age property is private,the __isset() method is called automatically.
1
Salin selepas log masuk
kaedah isset(), apabila Kaedah

unset() dipanggil apabila kaedah unset() dipanggil pada sifat yang tidak ditentukan atau tidak boleh diakses. Berikut ialah contoh:

Hasil larian kod adalah seperti berikut:

<?php
class Person
{
    public $sex;
    private $name;
    private $age;

    public function __construct($name="",  $age=25, $sex=&#39;Male&#39;)
    {
        $this->name = $name;
        $this->age  = $age;
        $this->sex  = $sex;
    }

    /**
     * @param $content
     *
     * @return bool
     */
    public function __unset($content) {
        echo "It is called automatically when we use the unset() method outside the class.<br>";
        echo  isset($this->$content);
    }
}

$person = new Person("John", 25); // Initially assigned.
unset($person->sex),"<br>";
unset($person->name),"<br>";
unset($person->age),"<br>";
Salin selepas log masuk
9 __sleep()

kaedah serialize() akan semak sama ada terdapat kaedah Magic __sleep(). Jika ada, kaedah ini akan dipanggil dahulu, dan kemudian operasi bersiri akan dilakukan. Kaedah
It is called automatically when we use the unset() method outside the class.
1
It is called automatically when we use the unset() method outside the class.
1
Salin selepas log masuk

__sleep() biasanya digunakan untuk menentukan atribut yang perlu disiri sebelum menyimpan data. Anda mungkin mendapati ciri ini berguna jika anda mempunyai beberapa objek yang sangat besar yang tidak semuanya perlu disimpan.

Untuk butiran, sila rujuk kod berikut:

Keputusan yang dijalankan adalah seperti berikut:

<?php
class Person
{
    public $sex;
    public $name;
    public $age;

    public function __construct($name="",  $age=25, $sex=&#39;Male&#39;)
    {
        $this->name = $name;
        $this->age  = $age;
        $this->sex  = $sex;
    }

    /**
     * @return array
     */
    public function __sleep() {
        echo "It is called when the serialize() method is called outside the class.<br>";
        $this->name = base64_encode($this->name);
        return array('name', 'age'); // It must return a value of which the elements are the name of the properties returned.
    }
}

$person = new Person('John'); // Initially assigned.
echo serialize($person);
echo '<br/>';
Salin selepas log masuk
__wakeup()

dengan Berbanding dengan kaedah
It is called when the serialize() method is called outside the class.
O:6:"Person":2:{s:4:"name";s:8:"5bCP5piO";s:3:"age";i:25;}
Salin selepas log masuk
sleep(), kaedah

wakeup() biasanya digunakan untuk operasi penyahserikatan, seperti mewujudkan semula sambungan pangkalan data atau melakukan operasi pemulaan lain.

Berikut ialah contoh yang berkaitan:

Keputusan yang dijalankan adalah seperti berikut:

<?php
class Person
{
    public $sex;
    public $name;
    public $age;

    public function __construct($name="",  $age=25, $sex=&#39;Male&#39;)
    {
        $this->name = $name;
        $this->age  = $age;
        $this->sex  = $sex;
    }

    /**
     * @return array
     */
    public function __sleep() {
        echo "It is called when the serialize() method is called outside the class.<br>";
        $this->name = base64_encode($this->name);
        return array('name', 'age'); // It must return a value of which the elements are the name of the properties returned.
    }

    /**
     * __wakeup
     */
    public function __wakeup() {
        echo "It is called when the unserialize() method is called outside the class.<br>";
        $this->name = 2;
        $this->sex = 'Male';
        // There is no need to return an array here.
    }
}

$person = new Person('John'); // Initially assigned.
var_dump(serialize($person));
var_dump(unserialize(serialize($person)));
Salin selepas log masuk
11 __toString()

Gunakan kaedah gema untuk mencetak terus Objek, kaedah __toString() akan dipanggil.
It is called when the serialize() method is called outside the class.
string(58) "O:6:"Person":2:{s:4:"name";s:8:"5bCP5piO";s:3:"age";i:25;}"
It is called when the unserialize() method is called outside the class.
object(Person)#2 (3) { ["sex"]=> string(3) "Male" ["name"]=> int(2) ["age"]=> int(25) }
Salin selepas log masuk

Nota: Kaedah ini mesti mengembalikan rentetan, jika tidak ralat maut akan dinaikkan pada tahap

. Dan anda tidak boleh membuang pengecualian dalam kaedah __toString() sama ada.

Berikut ialah contoh yang berkaitan:

E_RECOVERABLE_ERROR

Hasil menjalankan kod adalah seperti berikut:

<?php
class Person
{
    public $sex;
    public $name;
    public $age;

    public function __construct($name="",  $age=25, $sex=&#39;Male&#39;)
    {
        $this->name = $name;
        $this->age  = $age;
        $this->sex  = $sex;
    }

    public function __toString()
    {
        return  'go go go';
    }
}

$person = new Person('John'); // Initially assigned.
echo $person;
Salin selepas log masuk
Jadi, bagaimana jika kaedah __toString() ialah tidak ditakrifkan dalam kelas? Jom cuba.

go go go
Salin selepas log masuk
Hasil menjalankan kod adalah seperti berikut:

Catchable fatal error: Object of class Person could not be converted to string in D:\phpStudy\WWW\test\index.php on line 18
Salin selepas log masuk

显然,它在页面上报告了一个致命错误,PHP语法不支持这样的写法。

12. __invoke()

当您尝试以调用函数的方式调用对象时,__ invoke()方法将被自动调用。

注意:此功能仅在PHP 5.3.0及更高版本中有效。

下面是相关实例:

<?php
class Person
{
    public $sex;
    public $name;
    public $age;

    public function __construct($name="",  $age=25, $sex=&#39;Male&#39;)
    {
        $this->name = $name;
        $this->age  = $age;
        $this->sex  = $sex;
    }

    public function __invoke() {
        echo 'This is an object';
    }

}

$person = new Person('John'); // Initially assigned.
$person();
Salin selepas log masuk

运行代码结果如下:

This is an object
Salin selepas log masuk

如果坚持使用对象作为方法(但未定义__invoke()方法),则将得到以下结果:

Fatal error: Function name must be a string in D:\phpStudy\WWW\test\index.php on line 18
Salin selepas log masuk

13.__set_state()

从PHP 5.1.0开始,在调用var_export()导出类代码时会自动调用__set_state()方法。

__set_state()方法的参数是一个包含所有属性值的数组,其格式为array('property'=> value,...)

在以下示例中,我们没有定义__set_state()方法:

<?php
class Person
{
    public $sex;
    public $name;
    public $age;

    public function __construct($name="",  $age=25, $sex=&#39;Male&#39;)
    {
        $this->name = $name;
        $this->age  = $age;
        $this->sex  = $sex;
    }

}

$person = new Person('John'); // Initially assigned.
var_export($person);
Salin selepas log masuk

执行代码结果如下:

Person::__set_state(array( 'sex' => 'Male', 'name' => 'John', 'age' => 25, ))
Salin selepas log masuk

显然,对象的属性已打印。

现在让我们看看定义__set_state()方法的另一种情况:

<?php
class Person
{
    public $sex;
    public $name;
    public $age;

    public function __construct($name="",  $age=25, $sex=&#39;Male&#39;)
    {
        $this->name = $name;
        $this->age  = $age;
        $this->sex  = $sex;
    }

    public static function __set_state($an_array)
    {
        $a = new Person();
        $a->name = $an_array['name'];
        return $a;
    }

}

$person = new Person('John'); // Initially assigned.
$person->name = 'Jams';
var_export($person);
Salin selepas log masuk

执行代码结果如下:

Person::__set_state(array( 'sex' => 'Male', 'name' => 'Jams', 'age' => 25, ))
Salin selepas log masuk

14. __clone()

在PHP中,我们可以使用clone关键字通过以下语法克隆对象:

$copy_of_object = clone $object;
Salin selepas log masuk

但是,使用clone关键字只是一个浅拷贝,因为所有引用的属性仍将指向原始变量。

如果在对象中定义了clone()方法,则将在复制生成的对象中调用clone()方法,该方法可用于修改属性的值(如有必要)。

下面是相关的示例:

<?php
class Person
{
    public $sex;
    public $name;
    public $age;

    public function __construct($name="",  $age=25, $sex=&#39;Male&#39;)
    {
        $this->name = $name;
        $this->age  = $age;
        $this->sex  = $sex;
    }

    public function __clone()
    {
        echo __METHOD__."your are cloning the object.<br>";
    }

}

$person = new Person('John'); // Initially assigned.
$person2 = clone $person;

var_dump('persion1:');
var_dump($person);
echo '<br>';
var_dump('persion2:');
var_dump($person2);
Salin selepas log masuk

运行代码结果如下:

Person::__clone your are cloning the object.
string(9) "persion1:" object(Person)#1 (3) { ["sex"]=> string(3) "Male" ["name"]=> string(6) "John" ["age"]=> int(25) }
string(9) "persion2:" object(Person)#2 (3) { ["sex"]=> string(3) "Male" ["name"]=> string(6) "John" ["age"]=> int(25) }
Salin selepas log masuk

15.__autoload()

__autoload()方法可以尝试加载未定义的类。

过去,如果要在程序文件中创建100个对象,则必须使用include()或require()来包含100个类文件,或者必须在同一类文件中定义100个类。 例如以下:

/**
 * file non_autoload.php
 */

require_once('project/class/A.php');
require_once('project/class/B.php');
require_once('project/class/C.php');
.
.
.

if (ConditionA) {
    $a = new A();
    $b = new B();
    $c = new C();
    // …
} else if (ConditionB) {
    $a = newA();
    $b = new B();
    // …
}
Salin selepas log masuk

那么,如果我们使用__autoload()方法呢?

/**
 * file autoload_demo.php
 */
function  __autoload($className) {
    $filePath = “project/class/{$className}.php”;
    if (is_readable($filePath)) {
        require($filePath);
    }
}

if (ConditionA) {
    $a = new A();
    $b = new B();
    $c = new C();
    // …
} else if (ConditionB) {
    $a = newA();
    $b = new B();
    // …
}
Salin selepas log masuk

当PHP引擎第一次使用类A时,如果未找到类A,则autoload方法将被自动调用,并且类名称“ A”将作为参数传递。因此,我们在autoload()方法中需要做的是根据类名找到相应的类文件,然后将其包含在内。如果找不到该文件,则php引擎将抛出异常。

16. __debugInfo()

当执行 var_dump() 方法时,__debugInfo() 方法会被自动调用。如果 __debugInfo() 方法未被定义,那么 var_dump 方法或打印出这个对象的所有属性。

举例说明:

<?php
class C {
    private $prop;

    public function __construct($val) {
        $this->prop = $val;
    }

    /**
     * @return array
     */
    public function __debugInfo() {
        return [
            'propSquared' => $this->prop ** 2,
        ];
    }
}

var_dump(new C(42));
Salin selepas log masuk

执行结果:

object(C)#1 (1) { ["propSquared"]=> int(1764) }
Salin selepas log masuk

注意:__debugInfo() 方法应该在 PHP 5.6.0 及以上版本中使用。

总结

以上就是我所了解的 PHP 魔术方法,其中常用的包括 __set() 还有 __get()__autoload()。如果你还有其他疑问,可以从 PHP 官方网站获得更多帮助。

推荐学习:《PHP视频教程

Atas ialah kandungan terperinci Apakah kaedah ajaib php. Untuk maklumat lanjut, sila ikut artikel berkaitan lain di laman web China PHP!

Label berkaitan:
php
sumber:php.cn
Kenyataan Laman Web ini
Kandungan artikel ini disumbangkan secara sukarela oleh netizen, dan hak cipta adalah milik pengarang asal. Laman web ini tidak memikul tanggungjawab undang-undang yang sepadan. Jika anda menemui sebarang kandungan yang disyaki plagiarisme atau pelanggaran, sila hubungi admin@php.cn
Tutorial Popular
Lagi>
Muat turun terkini
Lagi>
kesan web
Kod sumber laman web
Bahan laman web
Templat hujung hadapan