PHP对象相关知识点的总结

转载请注明原文链接:http://www.cnblogs.com/XiongMaoMengNan/p/6674406.html

  • 对象传递:一种说法是“PHP对象是通过引用传递的”,更准确的说法是别名(标识符)传递,即它们都保存着同一个标识符(ID)的拷贝,这个标识符指向同一个对象的真正内容。

     1 <?php
     2 class A {
     3     public $foo = 1;
     4 }
     5
     6 $a = new A;
     7 $b = $a;     // $a ,$b都是同一个标识符的拷贝
     8              // ($a) = ($b) = <id>
     9 $b->foo = 2;
    10 echo $a->foo."\n";//2
    11
    12
    13 $c = new A;
    14 $d = &$c;    // $c ,$d是引用
    15              // ($c,$d) = <id>
    16
    17 $d->foo = 2;
    18 echo $c->foo."\n";//2
    19
    20
    21 $e = new A;
    22
    23 function foo($obj) {
    24     // ($obj) = ($e) = <id>
    25     $obj->foo = 2;
    26 }
    27
    28 foo($e);
    29 echo $e->foo."\n";//2
  • 对象复制:对象复制可以通过 clone 关键字来完成,如果原对象定义了 __clone() 方法,则新对象中的 __clone() 方法将在复制完后被调用,__clone() 方法可用于修改复制对象属性的值。当对象被复制后,会对对象的所有属性执行一个浅复制(shallow copy),但所有的引用属性仍然会是一个指向原来的变量的引用。

     1 <?php
     2 class SubObject
     3 {
     4     static $instances = 0;
     5     public $instance;
     6
     7     public function __construct()
     8     {
     9         $this->instance = ++self::$instances;
    10     }
    11
    12     public function __clone()
    13     {
    14         $this->instance = ++self::$instances;
    15     }
    16 }
    17
    18 class MyCloneable
    19 {
    20     public $object1;
    21     public $object2;
    22
    23     function __clone()
    24     {
    25         // 强制复制一份this->object, 否则仍然指向同一个对象
    26         $this->object1 = clone $this->object1;
    27     }
    28
    29     function cloneTest()
    30     {
    31         echo ‘cloneTest‘;
    32     }
    33 }
    34
    35 $obj = new MyCloneable();
    36
    37 $obj->object1 = new SubObject();
    38 $obj->object2 = new SubObject();
    39
    40 $obj2 = clone $obj;
    41
    42 print("Original Object:\n");
    43 print_r($obj);
    44
    45 print("Cloned Object:\n");
    46 print_r($obj2);
    47 echo $obj2->cloneTest().":\n";
    48 echo (new Reflectionclass($obj2));

    上列输出结果:

    Original Object:
    MyCloneable Object
    (
        [object1] => SubObject Object
            (
                [instance] => 1
            )
    
        [object2] => SubObject Object
            (
                [instance] => 2
            )
    
    )
    Cloned Object:
    MyCloneable Object
    (
        [object1] => SubObject Object
            (
                [instance] => 3
            )
    
        [object2] => SubObject Object
            (
                [instance] => 2
            )
    
    )
    cloneTest:
    Class [ <user> class MyCloneable ] {
      @@ /public/t.php 18-33
    
      - Constants [0] {
      }
    
      - Static properties [0] {
      }
    
      - Static methods [0] {
      }
    
      - Properties [2] {
        Property [ <default> public $object1 ]
        Property [ <default> public $object2 ]
      }
    
      - Methods [2] {
        Method [ <user> public method __clone ] {
          @@ /public/t.php 23 - 27
        }
    
        Method [ <user> public method cloneTest ] {
          @@ /public/t.php 29 - 32
        }
      }
    }
  • 对象遍历: foreach只能遍历对象的可见属性,无法遍历其方法,实现起来比较容易;另外,也可通过实现Iterator接口或IteratorAggregate接口的方法遍历对象属性。
  • 类型约束: PHP作为一种弱类型语言,类型约束可以让编程更加规范,也少出些差错;类型约束不只能用在对象定义中,也能用在函数定义中。类型约束可指定对象、接口、array、callable(闭包callback),类型约束用来保证实际数据类型与原型定义一致,不一致则抛出一个可捕获的致命错误;不过如果定义了默认值为NULL,那么实参可以是NULL;类型约束不能用于标量类型如 int 或 string,Traits 也不允许。
  • 对象序列化与还原:函数serialize()可将打成包含字节流的字符串便于存储对象,函数unserialize()能够还原字符串为对象。但有一个前提是,无论序列化还是反序列化,对象的类定义已经完成,即需要先导入类(文件)。
  • 重载:PHP的重载包括属性和方法,更像一个套用说法,不支持常见的重载语法规范,具有不可预见性,影响范围更宽泛,就是利用魔术方法(magic methods)来调用当前环境下未定义或不可见的类属性或方法。所有重载方法都必须被声明为 public(这一条应该比较好理解,别人可能因不可见才需要你,那你自己必须可见才行),参数也不能通过引用传递(重载方法具有不可预见性,估计出于安全方面的考虑吧,防止变量被随意引用)。在除 isset() 外的其它语言结构中无法使用重载的属性,这意味着当对一个重载的属性使用 empty() 时,重载魔术方法将不会被调用; 为避开此限制,必须将重载属性赋值到本地变量再使用 empty(),可见重载属性是介于合法属性与非法属性之间的存在。

    [属性重载]:这些方法不能被声明为 static,在静态方法中,这些魔术方法将不会被调用
    public void __set ( string $name , mixed $value )
    在给不可访问属性赋值时,__set() 会被调用
    
    public mixed __get ( string $name )
    读取不可访问属性的值时,__get() 会被调用
    
    public bool __isset ( string $name )
    当对不可访问属性调用 isset() 或 empty() 时,__isset() 会被调用
    
    public void __unset ( string $name )
    当对不可访问属性调用 unset() 时,__unset() 会被调用
    
    Note:
    因为 PHP 处理赋值运算的方式,__set() 的返回值将被忽略。类似的, 在下面这样的链式赋值中,__get() 不会被调用:
     $a = $obj->b = 8; 
    
    [方法重载]:
    public mixed __call ( string $name , array $arguments )
    在对象中调用一个不可访问方法时,__call() 会被调用
    
    public static mixed __callStatic ( string $name , array $arguments )
    在静态上下文中调用一个不可访问方法时,__callStatic() 会被调用
  • 静态属性方法:static 关键字用来定义静态属性、静态方法,静态属性不能通过实例化的对象-> 来访问(但静态方法可以)。静态属性只能被初始化为常量表达式,所以可以把静态属性初始化为整数或数组,但不能初始化为另一个变量或函数返回值,也不能指向一个对象。可以用一个变量表示类来动态调用静态属性,但该变量的值不能为关键字 self,parent 或 static。

     1 class Foo
     2 {
     3     public static $my_static = ‘foo‘;
     4
     5     public function staticValue() {
     6         return self::$my_static;
     7     }
     8 }
     9
    10 class Bar extends Foo
    11 {
    12     public function fooStatic() {
    13         return parent::$my_static;
    14     }
    15 }
    16
    17
    18 print Foo::$my_static . "\n";
    19
    20 $foo = new Foo();
    21 print $foo->staticValue() . "\n";
    22 print $foo->my_static . "\n";      // Undefined "Property" my_static
    23
    24 print $foo::$my_static . "\n";
    25 $classname = ‘Foo‘;
    26 print $classname::$my_static . "\n"; // As of PHP 5.3.0
    27
    28 print Bar::$my_static . "\n";
    29 $bar = new Bar();
    30 print $bar->fooStatic() . "\n";
  • 后期静态绑定:static:: 定义后期静态绑定工作原理是存储了上一个“非转发调用”(non-forwarding call)的类名。当进行静态方法调用时,该类名即为明确指定的那个(通常在 :: 运算符左侧部分);当进行非静态方法调用时,即为该对象所属的类。使用 self:: 或者 __CLASS__ 对当前类的静态引用,取决于定义当前方法所在的类;static:: 不再被解析为定义当前方法所在的类,而是在实际运行时计算的,可以用于静态属性和所有方法的调用。
     1 <?php
     2 class A
     3 {
     4
     5     private $proPrivate = "private of A";
     6     protected $proProtected = "protected of A";
     7     public $proPublic = "public of A";
     8
     9     private function foo()
    10     {
    11         echo $this->proPrivate."\n";
    12         echo $this->proProtected."\n";
    13         echo $this->proPublic."\n";
    14     }
    15
    16     public function test()
    17     {
    18         $this->foo();
    19         static::foo();
    20     }
    21 }
    22
    23 class B extends A
    24 {
    25    /* foo() will be copied to B, hence its scope will still be A and
    26     * the call be successful */
    27 }
    28
    29 class C extends A
    30 {
    31     private $proPrivate = "private of C";
    32     protected $proProtected = "protected of C";
    33     public $proPublic = "public of C";
    34
    35     private function foo()
    36     {
    37         /* original method is replaced; the scope of the new one is C */
    38         echo "I am C\n";
    39     }
    40
    41     public function myFoo()
    42     {
    43         //parent::foo();
    44         $this->foo();
    45     }
    46 }
    47
    48 echo "Class B:\n";
    49 $b = new B();
    50 $b->test();
    51 echo "\nClass C:\n";
    52 $c = new C();
    53 $c->myFoo();
    54 $c->test();   //fails

    上例输出结果:

    Class B:
    private of A
    protected of A
    public of A
    private of A
    protected of A
    public of A
    
    Class C:
    I am C
    private of A
    protected of C
    public of C
    Fatal error: Uncaught Error: Call to private method C::foo() from context ‘A‘ in /public/t.php:19 Stack trace: #0 /public/t.php(54): A->test() #1 {main} thrown in /public/t.php on line 19
  • 继承:官方文档对继承有这样一段描述“当扩展一个类,子类就会继承父类所有公有的和受保护的方法。除非子类覆盖了父类的方法,被继承的方法都会保留其原有功能”,言下之意似乎私有属性和方法不会被继承;然而上例又告诉我们子类拥有与父类一致的属性和方法,继承就是全盘复制,这才能满足我们对继承编程的需求,如果私有的不能继承,子类就必须自行重新定义,在大多数时候没有必要。另外就是可见性问题,父类的私有属性和方法在子类是不可见的。上例还告诉我们对象实际执行的域要考虑可见性、继承、后期静态绑定机制。

转载请注明原文链接:http://www.cnblogs.com/XiongMaoMengNan/p/6674406.html

时间: 2024-08-08 23:42:14

PHP对象相关知识点的总结的相关文章

js中闭包和对象相关知识点

学习js时候,读到几篇不错的博客.http://www.cnblogs.com/yexiaochai/p/3802681.html一,作用域 和C.C++.Java 等常见语言不同,JavaScript 的作用域不是以花括号包围的块级作用域(block scope),这个特性经常被大多数人忽视,因而导致莫名其妙的错误.例如下面代码,在大多数类C 的语言中会出现变量未定义的错误,而在JavaScript 中却完全合法: if(true) { var somevar = 'value'; } con

JAVA File对象相关知识点

public class DirList { public static void main(String[] args) { File file=new File("."); String[] list; if(args.length==0) { list=file.list(); }else { list=file.list(new DirFilter(args[0]));//new DirFilter出代码因为实现了FilenameFilter接口:所以只要实现了该接口的类都可以

hibernate框架的核心对象和相关知识点

Hibernate架构下图提供了hibernate体系的高层视图: Hibernate全面解决方案: Hibernate核心APIConfiguration负责管理数据库的配置信息.数据库的配置信息包含了Hibernate连接数据库的一些基本信息(hibernate.cfg.xml),Configuration对象可以通过加载配置文件获取并管理这些信息. Configuration config = new Configuration().configure();或者:File file = n

图像处理程序框架—MFC相关知识点

CDC:Windows使用与设备无关的图形设备环境(DC :Device Context) 进行显示 . MFC基础类库定义了设备环境对象类----CDC类.CDC与CGdiObject的关系 说道CDC类就不能不提一下GdiObject---图形对象类. 在Windows应用程序中,设备环境与图形对象共同工作,协同完成绘图显示工作.就像画家绘画一样,设备环境好比是画家的画布,图形对象好比是画家的画笔.用画笔在画布上绘画,不同的画笔将画出不同的画来.选择合适的图形对象和绘图对象,才能按照要求完成

Android开发涉及有点概念&amp;相关知识点(待写)

前言,承接之前的 IOS开发涉及有点概念&相关知识点,这次归纳的是Android开发相关,好废话不说了.. 先声明下,Android开发涉及概念比IOS杂很多,可能有很多都题不到的.. 首先由于Android是基于Linux,而Linux又是用c山寨Unix的,但是为什么Android不用开发App,其实我也不懂,我想可能是java比较简单. 同样的,IOS为啥没用java,这个我也想可能是因为IOS是封闭的吧..晕,又废话了..好,开始吧,不过这次很多要查资料了,很多名词.单词不会啊! 首先还

Python开发一个csv比较功能相关知识点汇总及demo

Python 2.7 doc demo: # -*- coding: utf-8 -*- import csv with open('eggs.csv', 'wb') as csvfile: spamwriter = csv.writer(csvfile, delimiter=' ', quotechar='|', quoting=csv.QUOTE_MINIMAL) spamwriter.writerow(['Spam'] * 5 + ['Baked Beans']) spamwriter.w

IOS之UI--小实例项目--添加商品和商品名(使用xib文件终结版) + xib相关知识点总结

*:first-child { margin-top: 0 !important; } body > *:last-child { margin-bottom: 0 !important; } a { color: #4183C4; } a.absent { color: #cc0000; } a.anchor { display: block; padding-left: 30px; margin-left: -30px; cursor: pointer; position: absolute

android点滴之触控相关知识点(持续更新)

1.检查当前屏幕有多少个触摸点. 使用MotionEvent的getPointerCount()方法. 2.多点触控下,触摸的状态的检测 MotionEvent类中方法getActionMasked(),这个方法可以检查触摸的状态: 当只有一个触摸点时(只用一根手指触摸屏幕),按下是0,起来是1,移动是2: 当触摸点多于1时(多根手指触摸屏幕),按下是5,起来是6,移动是2: 3.多点触控下,每个触点的分类处理 多点触摸时 假如是第i个点 int x = event.getX(i); int y

numeric_limits 模板的相关知识点

说白了,它是一个模板类,它主要是把C++当中的一些内建型别进行了封装,比如说numeric_limits<int>是一个特化后的类,从这个类的成员变量与成员函数中,我们可以了解到int的很多特性:可以表示的最大值,最小值,是否是精确的,是否是有符号等等.如果用其他任意(非内建类型)来特化这个模板类,比如string,string怎么可能有最大值?我们从MSDN上可以了解到,这对string,成员变量与成员函数是没有意义的,要么返回0要么为false. 具体的一些用法: #include <