PHP面向对象中的重要知识点(三)
1. namespace:
和C++中的名字空间很像,作用也一样,都是为了避免在引用较多第三方库时而带来的名字冲突问题。通过名字空间,即便两个class的名称相同,但是因为位于不同的名字空间内,他们仍然可以被精确定位和区分。第一次看到PHP的名字空间语法时,感觉和C++相比在语法上是非常非常相似的,然而在写点儿小例子做做实验的时候才发现,他们的差别还是很大的,为了避免以后忘记,所以这里特别将其记录了下来。见如下代码:
<?php
//in Test2.php
namespace nstest\test2; class Test2 {
public static function printMe() {
print 'This is nstest\test2\Test2::printSelf.'."\n";
}
} <?php
//in Test1.php
namespace nstest\test1; class Test1 {
public static function printMe() {
print 'This is nstest\test1\Test1::printSelf.'."\n";
}
}
require "Test2.php";
nstest\test2\Test2::printMe();
运行结果如下:
bogon:TestPhp$ php Test1.php
PHP Fatal error: Class 'nstest\test1\nstest\test2\Test2' not found in /Users/liulei/PhpstormProjects/TestPhp/Test1.php on line
是不是这个结果比较出乎意料,原因在哪呢?HOHO,原来PHP在进行名字空间引用的时候,如果名字空间的第一个字符不是前导斜杠(\),那么就被自动识别为相对名字空间,在上面的代码中,Test1自身所在的名字空间是namespace nstest\test1,因此在以nstest\test2\Test2::printMe()方式调用Test2::printMe()时,PHP将自动解析为nstest\test1\nstest\test2\Test2::printMe(),即认为nstest\test2是在当前名字空间内部的。修正该问题非常简单,只需在引用时加上前导斜杠(\)即可,见以下修复后的代码:
<?php
//Test2.php
namespace nstest\test2; class Test2 {
public static function printMe() {
print 'This is nstest\test2\Test2::printSelf.'."\n";
}
} <?php
//Test1.php
namespace nstest\test1; class Test1 {
public static function printMe() {
print 'This is nstest\test1\Test1::printSelf.'."\n";
}
}
require "Test2.php";
\nstest\test2\Test2::printMe();
运行结果如下:
bogon:TestPhp$ php Test1.php
This is nstest\test2\Test2::printSelf.
还有一种改动方式,可以示意一下PHP中名字空间中的相对引用。这里我们可以将Test1的名字空间改为namespace nstest,其他的修改见以下代码中红色高亮部分:
<?php
//Test2.php
namespace nstest\test2; class Test2 {
public static function printMe() {
print 'This is nstest\test2\Test2::printSelf.'."\n";
}
} <?php
//Test1.php
namespace nstest; class Test1 {
public static function printMe() {
print 'This is nstest\test1\Test1::printSelf.'."\n";
}
} require "Test2.php";
test2\Test2::printMe();
运行结果等于上面正确的结果。最重要的差别就是该例使用了PHP名字空间中的相对定位。相信熟悉C++的开发者一定会想到use关键字,PHP也提供了该关键字,他们的功能是一致的,都是为了避免在后面的代码中,无需再通过全限定符(类名前加名字空间前缀)来引用其他名字空间中的类了。至于具体的语法规则,还是看看下面具体的代码和关键性注释吧。
<?php
//Test2.php
namespace nstest\test2; class Test2 {
public static function printMe() {
print 'This is nstest\test2\Test2::printSelf.'."\n";
}
} <?php
//Test1.php
namespace nstest\test1; class Test1 {
public static function printMe() {
print 'This is nstest\test1\Test1::printSelf.'."\n";
}
} require "Test2.php";
//这里需要特别注意的是,nstest\test2已经表示名字空间绝对路径定位,不需要再加前导斜杠(\)了。
//另外这里还有一个隐式规则是test2表示该名字空间的缺省别名,在引用其名字空间内的对象时需要加test2前缀。
use nstest\test2;
test2\Test2::printMe(); //这里我们也可以给名字空间显式的指定别名,如:
use nstest\test2 as test2_alias;
test2_alias\Test2::printMe();
运行结果如下:
bogon:TestPhp$ php Test1.php
This is nstest\test2\Test2::printSelf.
This is nstest\test2\Test2::printSelf.
最后介绍一下PHP中全局名字空间的引用方式,见如下代码和关键性注释:
<?php
class Test {
public static function printMe() {
print 'This is Global namespace Test::printSelf.'."\n";
}
} //下面两行代码表示的是同一对象,即全局名字空间下的Test类,然而如果因为名字空间冲突导致第一种方式不能被PHP
//编译器正常识别,那么就可以使用第二种方式显式的通知PHP,自己要引用的是全局名字空间中的Test类。
Test::printMe();
\Test::printMe();
运行结果如下:
bogon:TestPhp$ php Test1.php
This is Global namespace Test::printSelf.
This is Global namespace Test::printSelf.
2. Reflection:
PHP中的反射和Java中java.lang.reflect包提供的功能一样,更有意思的是,就连很多方法命名和调用方式也是非常雷同的。他们都是由一些列可以分析类、类方法和方法参数的PHP内置类组成。我们这里主要介绍的是如下几个常用的内置类:(Reflection、RelectionClass、ReflectionMethod、ReflectionParameter和ReflectionProperty)。现在我们还是一步一步来理解,即从ReflectionClass开始给出示例代码和关键性注释:
<?php
class TestClass {
public $publicVariable; function publicMethod() {
print "This is publicMethod.\n";
}
} function classInfo(ReflectionClass $c) {
$details = "";
//getName将返回实际的类名。
$name = $c->getName();
if ($c->isUserDefined()) {
$details .= "$name is user defined.\n";
}
if ($c->isInternal()) {
$details .= "$name is built-in.\n";
}
if ($c->isAbstract()) {
$details .= "$name is abstract class.\n";
}
if ($c->isFinal()) {
$details .= "$name is final class.\n";
}
if ($c->isInstantiable()) {
$details .= "$name can be instantiated.\n";
} else {
$details .= "$name cannot be instantiated.\n";
}
return $details;
} function classSource(ReflectionClass $c) {
$path = $c->getFileName();
$lines = @file($path);
//获取类定义代码的起始行和截至行。
$from = $c->getStartLine();
$to = $c->getEndLine();
$len = $to - $from + 1;
return implode(array_slice($lines,$from - 1,$len));
} print "The following is Class Information.\n";
print classInfo(new ReflectionClass('TestClass')); print "\nThe following is Class Source.\n";
print classSource(new ReflectionClass('TestClass'));
运行结果如下:
bogon:TestPhp$ php reflection_test.php
The following is Class Information.
TestClass is user defined.
TestClass can be instantiated. The following is Class Source.
class TestClass {
public $publicVariable; function publicMethod() {
print "This is publicMethod.\n";
}
}
下面让我们仍然以代码示例和关键性注释的方法继续ReflectionMethod的学习之旅。
<?php
class TestClass {
public $publicVariable; function __construct() { }
private function privateMethod() { }
function publicMethod() {
print "This is publicMethod.\n";
}
function publicMethod2(string $arg1, int $arg2) { }
} //这个函数中使用的ReflectionMethod中的方法都是非常简单直观的,就不再过多赘述了。
function methodInfo(ReflectionMethod $m) {
$name = $m->getName();
$details = "";
if ($m->isUserDefined()) {
$details .= "$name is user defined.\n";
}
if ($m->isInternal()) {
$details .= "$name is built-in.\n";
}
if ($m->isAbstract()) {
$details .= "$name is abstract.\n";
}
if ($m->isPublic()) {
$details .= "$name is public.\n";
}
if ($m->isProtected()) {
$details .= "$name is protected.\n";
}
if ($m->isPrivate()) {
$details .= "$name is private.\n";
}
if ($m->isStatic()) {
$details .= "$name is static.\n";
}
if ($m->isFinal()) {
$details .= "$name is final.\n";
}
if ($m->isConstructor()) {
$details .= "$name is constructor.\n";
}
if ($m->returnsReference()) {
$details .= "$name returns a reference.\n";
}
return $details;
} function methodSource(ReflectionMethod $m) {
$path = $m->getFileName();
$lines = @file($path);
$from = $m->getStartLine();
$to = $m->getEndLine();
$len = $to - $from + 1;
return implode(array_slice($lines, $from - 1, $len));
} $rc = new ReflectionClass('TestClass');
$methods = $rc->getMethods();
print "The following is method information.\n";
foreach ($methods as $method) {
print methodInfo($method);
print "\n--------------------\n";
} print "The following is Method[TestClass::publicMethod] source.\n";
print methodSource($rc->getMethod('publicMethod'));
运行结果如下:
bogon:TestPhp$ php reflection_test.php
The following is method information.
__construct is user defined.
__construct is public.
__construct is constructor. --------------------
privateMethod is user defined.
privateMethod is private. --------------------
publicMethod is user defined.
publicMethod is public. --------------------
publicMethod2 is user defined.
publicMethod2 is public. --------------------
The following is Method[TestClass::publicMethod] source.
function publicMethod() {
print "This is publicMethod.\n";
}
让我们继续ReflectionParameter吧,他表示的是成员函数的参数信息。继续看代码吧。
<?php
class ParamClass { } class TestClass {
function publicMethod() {
print "This is publicMethod.\n";
}
function publicMethod2(ParamClass $arg1, &$arg2, $arg3 = null) { }
} function paramInfo(ReflectionParameter $p) {
$details = "";
//这里的$declaringClass将等于TestClass。
$declaringClass = $p->getDeclaringClass();
$name = $p->getName();
$class = $p->getClass();
$position = $p->getPosition();
$details .= "\$$name has position $position.\n";
if (!empty($class)) {
$classname = $class->getName();
$details .= "\$$name must be a $classname object\n";
}
if ($p->isPassedByReference()) {
$details .= "\$$name is passed by reference.\n";
}
if ($p->isDefaultValueAvailable()) {
$def = $p->getDefaultValue();
$details .= "\$$name has default: $def\n";
}
return $details;
} $rc = new ReflectionClass('TestClass');
$method = $rc->getMethod('publicMethod2');
$params = $method->getParameters(); foreach ($params as $p) {
print paramInfo($p)."\n";
}
运行结果如下:
bogon:TestPhp$ php reflection_test.php
$arg1 has position .
$arg1 must be a ParamClass object $arg2 has position .
$arg2 is passed by reference. $arg3 has position .
$arg3 has default:
上面介绍的都是通过PHP提供的Reflection API来遍历任意class的具体信息,事实上和Java等其他语言提供的反射功能一样,PHP也同样支持通过反射类调用实际对象的方法,这里将主要应用到两个方法,分别是ReflectionClass::newInstance()来创建对象实例,另一个是ReflectionMethod::invoke(),根据对象实例和方法名执行该方法。见如下代码:
<?php
class TestClass {
private $privateArg;
function __construct($arg) {
$this->privateArg = $arg;
}
function publicMethod() {
print '$privateArg = '.$this->privateArg."\n";
} function publicMethod2($arg1, $arg2) {
print '$arg1 = '.$arg1.' $arg2 = '.$arg2."\n";
}
} $rc = new ReflectionClass('TestClass');
$testObj = $rc->newInstanceArgs(array('This is private argument.'));
$method = $rc->getMethod('publicMethod');
$method->invoke($testObj); $method2 = $rc->getMethod('publicMethod2');
$method2->invoke($testObj,"hello","world");
运行结果如下:
bogon:TestPhp$ php reflection_test.php
$privateArg = This is private argument.
$arg1 = hello $arg2 = world
事实上ReflectionClass、ReflectionMethod和ReflectionParameter提供给我们的可用方法还有更多,这里只是给出几个最典型的方法,以便我们可以更为直观的学习和了解PHP Reflection API。相信再看完以后的代码示例之后,我们都会比较清楚,如果今后需要用到和class相关的功能,就从ReflectionClass中查找,而member function的信息则一定来自于ReflectionMethod,方法参数信息来自于ReflectionParameter。
注:该Blog中记录的知识点,是在我学习PHP的过程中,遇到的一些PHP和其他面向对象语言相比比较独特的地方,或者是对我本人而言确实需要簿记下来以备后查的知识点。虽然谈不上什么深度,但是还是希望能与大家分享。
PHP面向对象中的重要知识点(三)的更多相关文章
- PHP面向对象中的重要知识点(二)
1. __toString: 当对象被打印时,如果该类定义了该方法,则打印该方法的返回值,否则将按照PHP的缺省行为输出打印结果.该方法类似于Java中的toString(). <?php cl ...
- PHP面向对象中的重要知识点(一)
1. __construct: 内置构造函数,在对象被创建时自动调用.见如下代码: <?php class ConstructTest { private $arg1; private $arg ...
- 第35节:Java面向对象中的多线程
Java面向对象中的多线程 多线程 在Java面向对象中的多线程中,要理解多线程的知识点,首先要掌握什么是进程,什么是线程?为什么有多线程呢?多线程存在的意义有什么什么呢?线程的创建方式又有哪些?以及 ...
- [.net 面向对象程序设计进阶] (4) 正则表达式 (三) 表达式助手
[.net 面向对象程序设计进阶] (2) 正则表达式(三) 表达式助手 上面两节对正则表达式的使用及.NET下使用正则表达式作了详细说明,本节主要搜集整理了常用的正则表达式提供参考. 此外为了使用方 ...
- 文成小盆友python-num8 面向对象中的成员,成员修饰符,特殊成员,异常处理,设计模式之单例模式
本节主要内容: 1.面向对象中的成员 2.成员修饰符 3.特殊成员 4.异常处理 5.设计模式之单例模式 一.面向对象中的成员(类的成员) 类的成员总共可以分为3大类,每类中有不同的分支. 1.总述, ...
- JavaScript面向对象中的继承
1.1继承的基本概念 使用一个子类,继承另一个父类,那么子类可以自动拥有父类中的所有属性和方法,这个过程叫做继承. >>>继承的两方,发生在两个类之间. 实现继承的三种方式: 扩展O ...
- Python面向对象:杂七杂八的知识点
为什么有这篇"杂项"文章 实在是因为python中对象方面的内容太多.太乱.太杂,在写相关文章时比我所学过的几种语言都更让人"糟心",很多内容似独立内容.又似相 ...
- python基础知识点三
内置函数和匿名函数 python 一共有68个内置的函数:它们就是python提供给你直接可以拿来使用的所有函数 内置函数的图:链接 :https://www.processon.com/mindma ...
- 探讨 JS 的面向对象中继承的那些事
最近学了 JS 的面向对象,这篇文章主要是探讨 JS 的面向对象中继承的那些事. JS中继承的特点: 1.子类继承父类: 2.子类可以用父类的方法和属性 3.子类的改变可以不影响父类 下面用一个例子来 ...
随机推荐
- 关于Android的背景色配色小结
三基色原理:三基色是指红,绿,蓝三色,人眼对红.绿.蓝最为敏感,大多数的颜色可以通过红.绿.蓝三色按照不同的比例合成产生.同样绝大多数单色光也可以分解成红绿蓝三种色光.这是色度学的最基本原理,即三基色 ...
- java 多线程(threadlocal)
package com.example; import java.util.Random; public class App { public static class MyRunnable1 imp ...
- Html做三个平台原生APP啦
DeviceOne之前介绍过了,现在来介绍一下DeviceOne快速开发到什么程度 使用js只需要5分钟就可以打出垮Android.ios.windows三大平台的纯原生UI的安装包. 只需要6个小时 ...
- 【情人节来一发】网站添加QQ客服功能
今年的元宵节遇到情人节,挺不自量力的,呵呵,开篇给各位讲个段子,早上一美女同学在空间发说说道:“开工大吉 起床啦,卖元宵,卖玫瑰,卖避孕套啦-有木有一起去发财的小伙伴?Let’s go…”,对于此种长 ...
- hash_map的简洁实现
hash_map的简洁实现 hash_map是经常被使用的一种数据结构,而其实现方式也是多种多样.如果要求我们使用尽可能简单的方式实现hash_map,具体该如何做呢? 我们知道hash_map最 ...
- asp.net 验证码session为null的解决方案
最近在做Y集团的订单系统时,登陆页面在测试时发现一个以前没有注意到的问题,登陆页面需要使用验证码,引用了一个生成验证码的aspx页面,在aspx页面中生成session和验证码图片,在登陆页面的后台处 ...
- scikit-learn的梯度提升算法(Gradient Boosting)使用
前言:本文的目的是记录sklearn包中GBRT的使用,主要是官网各参数的意义:对于理论部分和实际的使用希望在只是给出出处,希望之后有时间能补充完整 摘要: 1.示例 2.模型主要参数 3.模型主要属 ...
- 爱上MVC~为DisplayNameFor添加扩展,支持PagedList集合
回到目录 DisplayNameFor方法是MVC提供给我们的,它可以将模型的DisplayName特性的值显示到页面上,这对程序员来说很是方便,在进行实体设计时,可以指定它的显示名称,然后MVC引擎 ...
- 大数据时代的IT架构设计
大数据时代的IT架构设计(来自互联网.银行等领域的一线架构师先进经验分享) IT架构设计研究组 编著 ISBN 978-7-121-22605-2 2014年4月出版 定价:49.00元 208页 ...
- Java的声明和访问介绍
1.类的声明 类本身的声明:对类的声明来说,主要包括类的访问权限声明和非访问修饰符的使用.对于一个普通的Java类(POJO)来说,主要的访问权限修饰符只有两个public和默认权限,内部类可以有pr ...