首页 >后端开发 >php教程 >利用PHP的作用域解析运算符(::)

利用PHP的作用域解析运算符(::)

不言
不言原创
2018-06-21 09:31:142183浏览

这篇文章主要介绍了关于利用PHP的作用域解析运算符(::),有着一定的参考价值,现在分享给大家,有需要的朋友可以参考一下

今天 看joomla源码的时候,才意识到。原来这个操作符还可以访问类的非静态方法啊。真的让我吃惊不好。一直以为作用域解析运算符只能访问类的static方法和static成员变量。

Scope Resolution Operator (::)
今天 看joomla源码的时候,才意识到。原来这个操作符还可以访问类的非静态方法啊。真的让我吃惊不好。一直以为作用域解析运算符只能访问类的static方法和static成员变量。
如果各位不相信,下面有个简单的小测试代码可以证明这个。 

class A{ 
private $_name = 'A'; 
function __construct(){ 
echo &#39;A construct <br />&#39;; 
} 
function test(){ 
echo &#39;A test() <br />&#39;; 
} 
} 
class B extends A{ 
private $_name = &#39;B&#39;; 
function __construct(){ 
parent::__construct(); 
echo &#39;B construct <br />&#39;; 
} 
function test(){ 
echo &#39;B test()&#39;; 
} 
} 
A::test(); 
echo &#39;######### <br />&#39;; 
B::test();

这段代码输入的结果为: 

A test() 
######### 
B test()

虽然A类中的test()和B类中的test都不是 static方法,但是一样可以用 “类名::方法名称(参数列表)” 的样式进行正确调用。他的效果和 new 一个类的实例,然后用这个实例调用
test方法是一个样的。
但是,如果我需要在test方法中打印name属性,直接用::来调用 会是怎么个情况那.我们首先来修改下 上面的代码。 

class A{ 
private $_name = &#39;A&#39;; 
function __construct(){ 
echo &#39;A construct <br />&#39;; 
} 
function test(){ 
echo &#39;A test() <br />&#39;, $this->$_name,&#39;<br />&#39;; 
} 
} 
class B extends A{ 
private $_name = &#39;B&#39;; 
function __construct(){ 
parent::__construct(); 
echo &#39;B construct <br />&#39;; 
} 
function test(){ 
echo &#39;B test()&#39;, $this->_name,&#39;<br />&#39;; 
} 
} 
A::test(); 
echo &#39;######### <br />&#39;; 
B::test();

上面的代码运行的结果 如下: 

Fatal error: Using $this when not in object context in D:\www\test\scoperefe.php on line 9 
[html]

那有的朋友就说了。你压根就没有实例化类A,当然不能直接用$this->_name的方式来访问成员变量$_name了,那么,是不是修改成self::$_name就行了哪?
说干就干,下面把上面的代码修改下 

[code] 
class A{ 
private $_name = &#39;A&#39;; 
function __construct(){ 
echo &#39;A construct <br />&#39;; 
} 
function test(){ 
echo &#39;A test() <br />&#39;, self::$_name,&#39;<br />&#39;; 
} 
} 
class B extends A{ 
private $_name = &#39;B&#39;; 
function __construct(){ 
parent::__construct(); 
echo &#39;B construct <br />&#39;; 
} 
function test(){ 
echo &#39;B test()&#39;, $this->_name,&#39;<br />&#39;; 
} 
} 
A::test(); 
echo &#39;######### <br />&#39;; 
B::test();

再运行上面的代码,结果如下: 

A test() Fatal error: Access to undeclared static property: A::$_name in D:\www\test\scoperefe.php on line 9

哦,原来不能用self 关键字访问当前类的非static方法。
现在,如果想正确的调用这个方法,有2个做法:
1、首先实例化类,然后用对象调用就可以直接使用$this->_name进行调用了;
2、将成员变量$_name设置为static;
上面的问题,相信大家都能够正确的处理。
其实我真正想说的是:
如果一个方法可以不进行实例化就调用,那么我们最好把这个方法使用static关键字修饰下。在实现方法的时候,只调用该类的static成员变量。这样就不会出现上面遇到问题了。
如果一个方法没有设置为static的方法。那么,最安全的做法还是用实例对象进行调用更为安全,因为,说不定什么时候就需要修改该方法的实现,在修改的时候,说不定就要调用该类中的
非static成员变量(因为,很大程度上在修改方法的实现的时候,已经忘记还有用类名直接调用这么一说)。
个人愚见。

以上就是本文的全部内容,希望对大家的学习有所帮助,更多相关内容请关注PHP中文网!

相关推荐:

关于PHP中Yii框架的组件行为的属性注入和方法注入

关于PHP自定义序列化接口Serializable的用法分析

以上是利用PHP的作用域解析运算符(::)的详细内容。更多信息请关注PHP中文网其他相关文章!

声明:
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn