1. __toString:

当对象被打印时,如果该类定义了该方法,则打印该方法的返回值,否则将按照PHP的缺省行为输出打印结果。该方法类似于Java中的toString()。

<?php
class TestClass {
public function __toString() {
return "This is TestClass::__toString.\n";
}
} $testObj = new TestClass();
print $testObj;

运行结果如下:

Stephens-Air:Desktop$ php Test.php
This is TestClass::__toString.

2. __get和__set:

这两个方法用于处理类中未声明的属性访问。当对象使用者试图访问未声明的对象属性时,__get()会被调用,并带有一个包含要访问的属性名称字符串作为参数。无论从__get()方法返回什么,都会直接返回给调用者,就如同带有该值的属性存在一样。另外需要注意的是,如果属性存在,但是其访问可见性为private或protected,那么这两个拦截方法同样会被调用,反之,如果属性存在切可访问,那么直接访问属性即可,这两个方法将不再会被调用。以下为__get()拦截方法的示例代码:

<?php
class TestClass {
private $privateField;
public $publicField;
public function __construct() {
$this->privateField = "This is a private Field.\n";
$this->publicField = "This is a public Field.\n";
} public function __get($property) {
print "__get() is called.\n";
$method = "get${property}";
if (method_exists($this, $method)) {
return $this->$method();
}
return "This is undefined field.\n";
}
public function getPrivateField() {
return $this->privateField;
}
} $testObj = new TestClass();
print $testObj->privateField;
print $testObj->undefinedField;
print $testObj->publicField;

运行结果如下:

Stephens-Air:Desktop$ php Test.php
__get() is called.
This is a private Field.
__get() is called.
This is undefined field.
This is a public Field.

__set()方法被调用的规则和__get()基本相同,差别是用于拦截未定义或不可见类属性的赋值操作。另外,该方法接收两个参数,分别是属性名称和要设定的值。见如下代码示例:

<?php
class TestClass {
private $privateField;
public $publicField;
public function __construct() {
$this->privateField = "This is a private Field.\n";
$this->publicField = "This is a public Field.\n";
}
public function __get($property) {
print "__get() is called.\n";
$method = "get${property}";
if (method_exists($this, $method)) {
return $this->$method();
}
return "This is an undefined field.\n";
}
public function __set($property, $value) {
print "__set is called.\n";
$method = "set${property}";
if (method_exists($this, $method)) {
$this->$method($value);
} else {
print "This is an undefined field.\n";
}
}
public function getPrivateField() {
return $this->privateField;
}
public function setPrivateField($value) {
$this->privateField = $value;
}
} $testObj = new TestClass();
$testObj->privateField = "This is a private Field after set.\n";
$testObj->undefinedField = "This is a undefined Field after set.\n";
$testObj->publicField = "This is a public Field after set.\n"; print $testObj->privateField;
print $testObj->undefinedField;
print $testObj->publicField;

运行结果如下:

Stephens-Air:Desktop$ php Test.php
__set is called.
__set is called.
This is an undefined field.
__get() is called.
This is a private Field after set.
__get() is called.
This is an undefined field.
This is a public Field after set.

3. __isset和__unset:

这两个拦截方法被调用的规则和__get()和__set()非常类似,只是用于类中不存在或不可见属性被isset()和unset()两个全局方法应用时才会被分别触发。

<?php
class TestClass {
private $privateField;
public $publicField;
public function __construct() {
$this->privateField = "Defined private field";
$this->publicField = "Defined public field";
}
public function __isset($property) {
print "__isset is called.\n";
return isset($this->$property);
}
public function __unset($property) {
print "__unset is called.\n";
if (isset($this->$property)) {
unset($this->$property);
}
}
} $testObj = new TestClass();
print 'isset($testObj->privateField) is '.(isset($testObj->privateField) ? "true" : "false")."\n";
print 'isset($testObj->undefinedField) is '.(isset($testObj->undefinedField) ? "true" : "false")."\n";
print 'isset($testObj->publicField) is '.(isset($testObj->publicField) ? "true" : "false")."\n"; print "After unset......\n";
//下面两个函数调用后,$testObj的两个对象属性均会变为不可用。
//另外从输出结果来看,__unset方法仅仅被调用一次,因为publicField为可见属性,所以__unset不会因该属性而被调用。
unset($testObj->privateField);
unset($testObj->publicField); print 'isset($testObj->privateField) is '.(isset($testObj->privateField) ? "true" : "false")."\n";
print 'isset($testObj->publicField) is '.(isset($testObj->publicField) ? "true" : "false")."\n";

运行结果如下:

Stephens-Air:Desktop$ php Test.php
__isset is called.
isset($testObj->privateField) is true
__isset is called.
isset($testObj->undefinedField) is false
isset($testObj->publicField) is true
After unset......
__unset is called.
__isset is called.
isset($testObj->privateField) is false
__isset is called.
isset($testObj->publicField) is false

4. __call:

__call()方法是一个非常有用但又非常容易被滥用的拦截方法。当对象使用者试图访问当前对象未定义的成员函数时,__call()会被自动调用,同时传递两个参数,分别为函数名称和传递给调用函数的所有参数(数组)。__call方法返回的任何值都会返回给函数调用者,就如同该成员函数真实存在一样。下面给出一个非常有用的委托示例。

<?php
class DelegateClass {
function printMessage($arg1, $arg2) {
print "DelegateClass:delegatedMethod is called.\n";
print '$arg1 = '.$arg1.'and $arg2 = '.$arg2."\n";
}
}
class TestClass {
private $delegateObj;
public function __construct() {
$this->delegateObj = new DelegateClass();
}
public function __call($method, $args) {
$this->delegateObj->$method($args[0],$args[1]);
}
} $testObj = new TestClass();
$testObj->printMessage("hello","world");

运行结果如下:

Stephens-Air:Desktop$ php Test.php
DelegateClass:delegatedMethod is called.
$arg1 = helloand $arg2 = world

从以上示例可以看出,TestClass并未声明printMessage成员方法,但是通过__call()方法的巧妙桥接直接传递给了委托对象。个人认为该技巧为双刃剑,切勿过度使用。

5. 回调函数:

回调函数的应用场景无须多述,在C/C++中充斥着无数的回调函数典型用例。 这里只是简单给出PHP中回调函数的使用规则。见如下示例代码和关键性注释:

<?php
class Product {
public $name;
public $price;
public function __construct($name, $price) {
$this->name = $name;
$this->price = $price;
}
} class ProcessSale {
private $callbacks;
function registerCallback($cb) {
if (!is_callable($cb)) {
throw new Exception("callback not callable.");
}
$this->callbacks[] = $cb;
}
function sale($product) {
print "{$product->name}: processing \n";
foreach ($this->callbacks as $cb) {
//以下两种调用方式均可。
call_user_func($cb, $product);
$cb($product);
}
}
} $logger = function($product) {
print " logging ({$product->name})\n";
}; $processor = new ProcessSale();
$processor->registerCallback($logger);
$processor->sale(new Product("shoes",6));
print "\n";
$processor->sale(new Product("coffee",6));

运行结果如下:

Stephens-Air:Desktop$ php Test.php
shoes: processing
logging (shoes)
logging (shoes) coffee: processing
logging (coffee)
logging (coffee)

6. use(闭包):

在Javascript中存在大量的闭包应用,PHP中的闭包则是通过use关键字来完成的。对于闭包这个概念本身而言,简要的说就是函数内的代码可以访问其父作用域中的变量。见如下示例代码和关键性注释:

<?php
class Product {
public $name;
public $price;
public function __construct($name, $price) {
$this->name = $name;
$this->price = $price;
}
} class ProcessSale {
private $callbacks;
function registerCallback($cb) {
if (!is_callable($cb)) {
throw new Exception("callback not callable.");
}
$this->callbacks[] = $cb;
}
function sale($product) {
print "{$product->name}: processing \n";
foreach ($this->callbacks as $cb) {
$cb($product);
}
}
} class Totalizer {
static function warnAmount($amt) {
$count = 0;
//注意这里的$amt和$count均为闭包变量,其中&$count是以引用的形式传递的,即一旦函数内部修改了该变量的值,
//那么下次再访问该闭包变量时,$count将为之前调用中修改后的值。
return function($product) use($amt, &$count) {
$count += $product->price;
print " count: $count\n";
if ($count > $amt) {
print " high price reached: {$count}\n";
}
};
}
} $processor = new ProcessSale();
$processor->registerCallback(Totalizer::warnAmount(8));
$processor->sale(new Product("shoes",6));
$processor->sale(new Product("coffee",6));

运行结果如下:

shoes: processing
count:
coffee: processing
count:
high price reached:

注:该Blog中记录的知识点,是在我学习PHP的过程中,遇到的一些PHP和其他面向对象语言相比比较独特的地方,或者是对我本人而言确实需要簿记下来以备后查的知识点。虽然谈不上什么深度,但是还是希望能与大家分享。

PHP面向对象中的重要知识点(二)的更多相关文章

  1. PHP面向对象中的重要知识点(三)

    1. namespace: 和C++中的名字空间很像,作用也一样,都是为了避免在引用较多第三方库时而带来的名字冲突问题.通过名字空间,即便两个class的名称相同,但是因为位于不同的名字空间内,他们仍 ...

  2. PHP面向对象中的重要知识点(一)

    1. __construct: 内置构造函数,在对象被创建时自动调用.见如下代码: <?php class ConstructTest { private $arg1; private $arg ...

  3. [.net 面向对象程序设计进阶] (3) 正则表达式 (二) 高级应用

    [.net 面向对象程序设计进阶] (2) 正则表达式 (二)  高级应用 上一节我们说到了C#使用正则表达式的几种方法(Replace,Match,Matches,IsMatch,Split等),还 ...

  4. 文成小盆友python-num8 面向对象中的成员,成员修饰符,特殊成员,异常处理,设计模式之单例模式

    本节主要内容: 1.面向对象中的成员 2.成员修饰符 3.特殊成员 4.异常处理 5.设计模式之单例模式 一.面向对象中的成员(类的成员) 类的成员总共可以分为3大类,每类中有不同的分支. 1.总述, ...

  5. Python面向对象:杂七杂八的知识点

    为什么有这篇"杂项"文章 实在是因为python中对象方面的内容太多.太乱.太杂,在写相关文章时比我所学过的几种语言都更让人"糟心",很多内容似独立内容.又似相 ...

  6. 第35节:Java面向对象中的多线程

    Java面向对象中的多线程 多线程 在Java面向对象中的多线程中,要理解多线程的知识点,首先要掌握什么是进程,什么是线程?为什么有多线程呢?多线程存在的意义有什么什么呢?线程的创建方式又有哪些?以及 ...

  7. 20155322 2016-2017-2 《Java面向对象程序设计》第十二周课堂练习之Arrays和String单元测试

    20155322 2016-2017-2 <Java面向对象程序设计>第十二周课堂练习之Arrays和String单元测试 练习目地 在IDEA中以TDD的方式对String类和Array ...

  8. 探讨 JS 的面向对象中继承的那些事

    最近学了 JS 的面向对象,这篇文章主要是探讨 JS 的面向对象中继承的那些事. JS中继承的特点: 1.子类继承父类: 2.子类可以用父类的方法和属性 3.子类的改变可以不影响父类 下面用一个例子来 ...

  9. 面向对象中特殊方法的补充、isinstance/issubclass/type、方法和函数、反射

    一.面向对象中特殊方法的补充 1.__str__ 能将对象名改成你想要的字符串,但是类型还是类 class Foo(object): def __init__(self): pass def func ...

随机推荐

  1. C++模板&泛型编程

    ---恢复内容开始--- 一.泛型编程 定义:编写与类型无关的逻辑代码,是代码复用的一种手段.模板是泛型编程的基础 模板分为:函数模板和类模板 函数模板:代表了一个函数家族,该函数与类型无关,在使用时 ...

  2. Spring声明式事务管理基于tx/aop命名空间

    目的:通过Spring AOP 实现Spring声明式事务管理; Spring支持编程式事务管理和声明式事务管理两种方式. 而声明式事务管理也有两种常用的方式,一种是基于tx/aop命名空间的xml配 ...

  3. (转载)JAVA敏捷开发环境搭建

    整个软件项目分为四个环境 开发本地环境.开发环境.测试环境.IDC环境.和传统C++开发不一样的模式是多了第一个开发本地环境.这是为什么呢,因为目前大部分开发人员还是比较熟悉windows下开发.对于 ...

  4. Apache Lucene 4.5 发布,Java 搜索引擎

    Apache Lucene 4.5 发布了,该版本提供基于磁盘的文档值以及改进了过滤器的缓存.Lucene 4.5 的文档请看这里. Lucene 是apache软件基金会一个开放源代码的全文检索引擎 ...

  5. Android性能优化典范第二季

      Google前几天刚发布了Android性能优化典范第2季的课程,一共20个短视频,包括的内容大致有:电量优化,网络优化,Wear上如何做优化,使用对象池来提高效率,LRU Cache,Bitma ...

  6. 关于MySQL的在线扩容

    原文地址:http://bucketli.iteye.com/blog/1294032 主要简单总结下,mysql在线扩容和缩容一般涉及到的内容,主要包括三个方面,1.在线也就意味着需要把增量的数据重 ...

  7. 基础调试命令 - wt (watch and trace)

    本文介绍windbg动态调试过程中一个非常有用的命令,wt的用法. wt命令 wt命令之所以称为wt是因为它是watch and trace的简称,即用来观察和跟踪的命令.这个命令一般用在动态调试而不 ...

  8. Visual Studio 2015速递(2)——提升效率和质量(VS2015核心竞争力)

    系列文章 Visual Studio 2015速递(1)——C#6.0新特性怎么用 Visual Studio 2015速递(2)——提升效率和质量(VS2015核心竞争力) Visual Studi ...

  9. C++ const && define

    本文记录了C++中的const关键字的内容,分为3个部分,const和define的区别,const的作用,const的使用. const和define的区别 const的作用 const用于定义常量 ...

  10. scikit-learn的梯度提升算法(Gradient Boosting)使用

    前言:本文的目的是记录sklearn包中GBRT的使用,主要是官网各参数的意义:对于理论部分和实际的使用希望在只是给出出处,希望之后有时间能补充完整 摘要: 1.示例 2.模型主要参数 3.模型主要属 ...