设为首页 收藏本站
查看: 866|回复: 0

[经验分享] Php魔术函数学习与应用 __construct() __destruct() __get()等

[复制链接]

尚未签到

发表于 2017-4-8 10:05:35 | 显示全部楼层 |阅读模式
  (1)初识魔术方法

Php5.0发布以来为我们提供了很多面向对象

的特性,尤其是为我们提供了好多易用的魔术方法,这些魔术方法可以让我们简化我们的编码,更好的设计我们的系统。今天我们就来认识下php5.0给我们提供的魔术方法。


1,__construct() 当实例化一个对象的时候,这个对象的这个方法首先被调用。



class Test

{

function __construct()


{

echo "before";

}

}



$t = new Test();



输出是:


start
  <?

class Test

{

    function Test()

{

echo "end2";

}

function __construct()

{

echo "end";

}


}

$t = new Test();
  //输出end

?>

我们知道php5对象模型
和类名相同
函数
是类的构造函数
,那么如果我们同时定义构造函数
和__construct()方法的话,php5会默认调用构造函数
而不会调用同类名函数
,所以__construct()作为类的默认的构造函数




2,__destruct() 当删除一个对象或对象操作终止的时候,调用该方法。



class Test

{

function __destruct()

{

echo "end";

}

}

$t = new Test();

将会输出

end



我们就可以在对象操作结束的时候进行释放资源之类的操作


3,__get() 当试图读取一个并不存在的属性的时候被调用。

如果试图读取一个对象并不存在的属性的时候,PHP就会给出错误信息。如果在类里添加__get方法,并且我们可以用这个函数
实现类似java中反射的各种操作。



class Test

{

public function __get($key)

{

echo $key . " 不存在";

}

}



$t = new Test();

echo $t->name;


就会输出:

name 不存在



4,__set() 当试图向一个并不存在的属性写入值的时候被调用。



class Test

{

public function __set($key,$value)

{

echo '对'.$key . "附值".$value;

}

}



$t = new Test();

$t->name = "aninggo";


就会输出:

对 name 附值 aninggo



5,__call() 当试图调用一个对象并不存在的方法时,调用该方法。

class Test

{

public function __call($Key, $Args)

{

echo "您要调用的 {$Key} 方法不存在。你传入的参数是:" . print_r($Args, true);

}

}


$t = new Test();

$t->getName(aning,go);




程序将会输出:

您要调用的 getName 方法不存在。参数是:Array

(

[0] => aning

[1] => go

)


6,__toString() 当打印一个对象的时候被调用

这个方法类似于java的toString方法,当我们直接打印对象的时候回调用这个函数


class Test

{

public function __toString()

{

return "打印 Test";

}

}



$t = new Test();


echo $t;






运行echo $t;的时候,就会调用$t->__toString();从而输出

打印 Test


7,__clone() 当对象被克隆时,被调用

class Test

{


public function __clone()

{

echo "我被复制了!";

}

}


$t = new Test();

$t1 = clone $t;


程序输出:

我被克隆了!

  __sleep 和 __wakeup

  串行化serialize可以把变量包括对象,转化成连续bytes数据. 你可以将串行化后的变量存在一个文件里或在网络上传输.
然后再反串行化还原为原来的数据. 你在反串行化类的对象之前定义的类,PHP可以成功地存储其对象的属性和方法.
有时你可能需要一个对象在反串行化后立即执行. 为了这样的目的,PHP会自动寻找__sleep和__wakeup方法.


  当一个对象被串行化,PHP会调用__sleep方法(如果存在的话). 在反串行化一个对象后,PHP 会调用__wakeup方法.
这两个方法都不接受参数. __sleep方法必须返回一个数组,包含需要串行化的属性. PHP会抛弃其它属性的值.
如果没有__sleep方法,PHP将保存所有属性.


  例子6.16显示了如何用__sleep和__wakeup方法来串行化一个对象. Id属性是一个不打算保留在对象中的临时属性.
__sleep方法保证在串行化的对象中不包含id属性. 当反串行化一个User对象,__wakeup方法建立id属性的新值.
这个例子被设计成自我保持. 在实际开发中,你可能发现包含资源(如图像或数据流)的对象需要这些方法


Object serialization

CODE: [Copy to clipboard]

--------------------------------------------------------------------------------


<?php


class User

{

public $name;

public $id;


function __construct()

{

//give user a unique ID 赋予一个不同的ID

$this->id = uniqid();

}


function __sleep()

{

//do not serialize this->id 不串行化id

return(array("name"));

}


function __wakeup()

{

//give user a unique ID

$this->id = uniqid();

}

}


//create object 建立一个对象

$u = new User;

$u->name = "Leon";


//serialize it 串行化 注意不串行化id属性,id的值被抛弃

$s = serialize($u);


//unserialize it 反串行化 id被重新赋值

$u2 = unserialize($s);


//$u and $u2 have different IDs $u和$u2有不同的ID

print_r($u);

print_r($u2);

?>

__set_state and __invoke
  测试代码如下:


1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20


<?php
 
class
A {
public
static function
__set_state(
$args
)
{
$obj
=
new
A(
)
;
foreach
(
$args
as
$k
=>
$v
)
{
$obj
->
$k
=
$v
;
}
return
$obj
;
}
}
 
$a
=
new
A;
$a
->
name
=
'cluries'
;
$a
->
sex
=
'female'
;
 
eval
(
'$b = '
.
var_export
(
$a
,
true
)
.
';'
)
;
 
print_r
(
$b
)
;
?>



  程序输出

object(A)#2 (2) {
["name"]=>  string(7) "cluries"
["sex"]=>  string(6) "female"
}

  得出以下结论,__set_state作用是用来复制一个对象,并且可以在__set_state中定义在复制对象的时候对复制得到的对象进行一些
改变。和__clone不同的是__set_state可以接受参数,__set_state使用起来更加强大!虽然个人觉得这个东西不是很好用= =!
  然后再说下__invoke:

手册上有个非常显眼的:Note: This feature is available since PHP 5.3.0. 


The __invoke method is called when a script tries to call an object as a function.

  __invoke方法将会在代码试图把对象当作函数来使用时候调用?有点稀奇,这个功能有什么用处呢?

然后看下提供的例子:


1
2
3
4
5
6
7
8
9
10


<?php
class
CallableClass {
function
__invoke(
$x
)
{
var_dump
(
$x
)
;
}
}
$obj
=
new
CallableClass;
$obj
(
5
)
;
var_dump
(
is_callable
(
$obj
)
)
;
?>



  程序输出:

int(5)
bool(true)

  还真是把对象当函数使用…
  __autoload

  PHP5中有一方法: __autoload() , 简单的说就是类的自动加载;


当你尝试使用一个PHP没有组织到的类, 它会寻找一个__autoload的全局函数. 如果存在这个函数,PHP会用一个参数来调用它,参数即类的名称。
  那么简单测试一下。
  

首先建一个名为”Test_autoload.php”的文件:



  • < ?
    php

  • /**

  • * 测试__autoload方法

  • *

  • */

  • class

    Test_autoload

    {

  • public

    function

    __construct
    ()

    {

  • echo

    "
    Test_autoload.
    "
    ;

  • }

  • }

  • ?>


  注意类名哦
, 然后随便建个文件重写 __autoload() 方法,这里假设是”test.php”;



  • < ?
    php

  • /**

  • * 重写 __autoload方法

  • */

  • function

    __autoload
    (
    $class
    )

    {

  • include

    $class
    .
    '
    .php
    '
    ;

  • }


  • $test
    =
    new

    Test_autoload
    ()
    ;

  • unset
    (
    $test
    )
    ;

  • ?>



  最后结果为:Test_autoload.



  

------------------------------------------------------------------------------------

8.顺便介绍下php5中提供的几个非常COOl的实验性函数


(1)。runkit_method_rename

     这个函数
可以动态的改变我们所调用的函数

名字




class Test

{


function foo() {

         return "foo! ";

     }


}


runkit_method_rename(

     'Test', //类名

     'foo',//实际调用的函数


     'bar'//显示调用的函数


);


echo Test::bar();


程序将输出

 

foo!




(2) runkit_method_add


这个函数
可以动态的向类中添加函数




class Test

{


function foo() {

         return "foo! ";

     }


}


runkit_method_add(

     Test,  //类名

     'add', //新函数


     '$num1, $num2',//传入参数

     'return $num1 + $num2;',//执行的代码

     RUNKIT_ACC_PUBLIC

);


// 调用

echo $e->add(12, 4);




(3)runkit_method_copy

可以把A类中的函数
拷贝到类B中并对函数
重命名



class Foo {

     function example() {

         return "foo! ";

     }

}


class Bar {

     //空类

}


//执行拷贝

runkit_method_copy('Bar', 'baz', 'Foo', 'example');


//执行拷贝后的函数


echo Bar::baz();




(4) runkit_method_redefine

动态的修改函数
的返回值

这个函数
可以让我们轻松的实现对类的MOCK测试!是不是很COOL呢



class Example {

     function foo() {

         return "foo! ";

     }

}


//创建一个测试对象

$e = new Example();


// 在测试对象之前输出

echo "Before: " . $e->foo();


// 修改返回值

runkit_method_redefine(

     'Example',

     'foo',

     '',

     'return "bar! ";',

     RUNKIT_ACC_PUBLIC

);


// 执行输出

echo "After: " . $e->foo();




(5)runkit_method_remove

这个函数
就很简单了,看名字
就能看出来了,动态的从类中移除函数


class Test {

     function foo() {

         return "foo! ";

     }

   

     function bar() {

         return "bar! ";

     }

}


// 移除foo函数


runkit_method_remove(

     'Test',

     'foo'

);


echo implode(' ', get_class_methods('Test'));


程序输出

bar

运维网声明 1、欢迎大家加入本站运维交流群:群②:261659950 群⑤:202807635 群⑦870801961 群⑧679858003
2、本站所有主题由该帖子作者发表,该帖子作者与运维网享有帖子相关版权
3、所有作品的著作权均归原作者享有,请您和我们一样尊重他人的著作权等合法权益。如果您对作品感到满意,请购买正版
4、禁止制作、复制、发布和传播具有反动、淫秽、色情、暴力、凶杀等内容的信息,一经发现立即删除。若您因此触犯法律,一切后果自负,我们对此不承担任何责任
5、所有资源均系网友上传或者通过网络收集,我们仅提供一个展示、介绍、观摩学习的平台,我们不对其内容的准确性、可靠性、正当性、安全性、合法性等负责,亦不承担任何法律责任
6、所有作品仅供您个人学习、研究或欣赏,不得用于商业或者其他用途,否则,一切后果均由您自己承担,我们对此不承担任何法律责任
7、如涉及侵犯版权等问题,请您及时通知我们,我们将立即采取措施予以解决
8、联系人Email:admin@iyunv.com 网址:www.yunweiku.com

所有资源均系网友上传或者通过网络收集,我们仅提供一个展示、介绍、观摩学习的平台,我们不对其承担任何法律责任,如涉及侵犯版权等问题,请您及时通知我们,我们将立即处理,联系人Email:kefu@iyunv.com,QQ:1061981298 本贴地址:https://www.yunweiku.com/thread-361818-1-1.html 上篇帖子: [转]利用 PHP 的 SPL 快速实现 Observer 设计模式 下篇帖子: PHP中文乱码之开发中的常见问题
您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

扫码加入运维网微信交流群X

扫码加入运维网微信交流群

扫描二维码加入运维网微信交流群,最新一手资源尽在官方微信交流群!快快加入我们吧...

扫描微信二维码查看详情

客服E-mail:kefu@iyunv.com 客服QQ:1061981298


QQ群⑦:运维网交流群⑦ QQ群⑧:运维网交流群⑧ k8s群:运维网kubernetes交流群


提醒:禁止发布任何违反国家法律、法规的言论与图片等内容;本站内容均来自个人观点与网络等信息,非本站认同之观点.


本站大部分资源是网友从网上搜集分享而来,其版权均归原作者及其网站所有,我们尊重他人的合法权益,如有内容侵犯您的合法权益,请及时与我们联系进行核实删除!



合作伙伴: 青云cloud

快速回复 返回顶部 返回列表