`
bardo
  • 浏览: 372958 次
  • 性别: Icon_minigender_1
  • 来自: 上海
博客专栏
D1407912-ab64-3e76-ae37-b31aa4afa398
浅述PHP设计模式
浏览量:11648
9d6df9f7-91da-3787-a37c-0e826525dd5d
Zend Framewor...
浏览量:9999
85b628bd-a2ed-3de2-a4b1-0d34985ae8b6
PHP的IDE(集成开发环...
浏览量:9359
社区版块
存档分类
最新评论

php中绕人的类成员访问控制

    博客分类:
  • PHP
阅读更多

        PHP5面向对象增加了类成员的访问控制。但是,这一点,表面上易于理解,但很多人实际是并不理解的。究其原因,主要是没有能够明白访问控制是如何定义的。

        首先,PHP定义访问控制与传统面向对象一样,定立了三个分级权限,Public:公有,Protected:保护,Private:私有。同时,还有静态成员,实际是应当算是四个权限。或者说是六种组合。

        先分别介绍一下三个分级权限:

        Public:公有——实际就是开放的,只要是在应用中,任一地方,通过对象的实例,均可以对其访问。

        Protected:保护——仅对子类开放。这就是说,只有两个区域中能够访问到:其一,当前类的成员方法当中的代码。其二,通过继承当前类生成的子类中成员方法的代码。其它地方均是不可访问的。

        Private:私有——说白了,就是自己用的。这就是说只有当前类的成员方法中的代码可以访问。

        这里要注意一下:我说的是:“当前类的成员方法中的代码”。因此,并不限是普通方法,还是静态方法。所以,你在静态方法中创建一个当前类的实例,那么,此实例是能够访问到该私有方法的。

        为什么要这么做?这实际上是必须的。比如,设计模式中的单件模式,它是将构造函数私有化,然后在通常名为getInstance()的方法中判断此类有没有实例化。保证该类只有单例存在。

        所以,无论是:实例名->私有方法,还是$this->私有方法,均是可以访问的。但“实例名->私有方法”必须是在类的成员方法之中。这就是说:私有的访问代码只要求是在当前类的class{}之中,而不是在外面就可以访问。

        不妨再看看单例的实现代码:

       

class phpApp{
	static private $instance;
	private $name;
	private __contruct($name){
		$this->name=$name;
	}
	static public getInstance($name){
		if (!isset(self::$instance){
			//这里实际是访问的私有的构造函数
			self::$instance=new phpApp($name);
		}
		return self::$instance;
	}
}

         再说静态,静态实际上是,把数据或方法交给类。由类来掌管。上面单例模式中,由类来掌管的有两个,第一是保存类实例的变量:$instance,因为只能有一个,所以,由类来管理最合适。同样,getInstance方法也是,它要保证无论是否实例化,均能访问到。

        实际开发中,我们有时,不知此方法或此属性交给类好,还是交给实例好。不妨这样想一下,这仍是访问控制。为什么这么说呢?只要是交给类管理的静态属性或方法,那么,你的应用中永远只有一个。

        假如说,你做一个链表。那么,每一个节点都是一个类。链表中有一项属性,是配置数据$config,很多初学者就不清楚它要定义成静态的(static)。这就是一个基本常识错误。想一想,不管有多少节点,节点创建,或其中函数运行时,要用到的配置,当且仅当只需要一个,这就足够了。如果其配置仅希望构造函数时传入,则就可以定义为: static private $config, 也许,你会说:这样问题来了。我仍每次创建子节点时要传入配置呀。其实不是这样的。唯有根节点传入一次即可。这就是说:将这个参数设为可选参数,如果传入了,就保存,否则,就不保存。代码如下:

      

class node{
	static private $config;
	private $name;
	private __contruct($config = NULL){
		if ($config != NULL)
		self::$config=$config;
	}	
}

 

        相当简单吧。但如果没有弄明白时,可能还是不知怎么做。每个节点都保存配置,既浪费时间,也浪费空间!

3
1
分享到:
评论
6 楼 ppxz2008 2011-08-10  
单列模式
必须加上:final private function __clone(){}
防止通过引用,创建实例对象!
5 楼 bardo 2011-04-01  
hnxxwyq 写道

$phpApp = phpApp::getInstance();
$phpApp1 = unserialize(serialize($phpApp));

这又变成2个对象了!

没错,这里也没有序列化保护。不过,我觉得,单例用得最多的是聚合复用,或者是应用资源共享。从这一点来说,一般不需要序列化保护。如果真的哪个傻仅会把单例序列化,那么,应当考虑,这个单例是不是滥用?
内部防犯自然是必不可少,但换一种模式呢?比如:注册表模式?情况会有所不同吧?
4 楼 hnxxwyq 2011-04-01  

$phpApp = phpApp::getInstance();
$phpApp1 = unserialize(serialize($phpApp));

这又变成2个对象了!
3 楼 115300111 2011-03-28  
明白了
2 楼 bardo 2011-03-26  
你说得完全正确,加10分。
正确的单例,必须要加上:
final private function __clone(){}
1 楼 rtxbc 2011-03-26  
你写的单例模式,我可以通过clone操作,创建两个对象。

相关推荐

Global site tag (gtag.js) - Google Analytics