复制代码
程序运行结果为:fatal error: call to private a::__construct() from invalid context in e:\phpprojects\test.php on line 6
我们已经禁止外部用new实例化这个类,我们改如何让用户访问这个类呢?前门堵了,我们需要给用户留个后门。解决办法是:static 修饰的方法,可以不经实例化一个类就可以直接访问这个方法。//不能用new实例化的类.
//static的方法留给外部访问.
//在方法内部返回实例.
class a{ private function __construct(){} static function getclassa(){ $a = new a(); return $a; } }// 看到这里确实返回的是 a 的实例.但不是同一个对象.$a1 = a::getclassa();$a2 = a::getclassa();echo \$a1 的类是 .get_class($a1). , \$a2 是 .get_class($a1);if($a1 === $a2){ echo
\$a1 \$a2 指向同一对象.;}else{ echo
\$a1 \$a2 不是一个对象.;}?>
复制代码
程序运行结果为:$a1 的类是 a , $a2 是 a$a1 $a2 不是一个对象.
我们已经通过static方法返回了a的实例。但还有问题。我们如何保证我们多次操作获得的是同一个实例的呢?
解决办法:static的属性在内部也只有一个。static 属性能有效的被静态方法调用。将这个属性也设置成private,以防止外部调用。先将这个属性设置成 null。每次返回对象前,先判断这个属性是否为 null 。如果为 null 就创建这个类的新实例,并赋值给这个 static 属性。如果不为空,就返回这个指向实例的 static 属性。//不能用new实例化的类.
//static的方法留给外部访问.
//在方法内部返回实例.
//定义静态属性保证这个实例能被静态方法调用.
//增加判断部分.
class a{ private static $a = null; private function __construct(){} static function getclassa(){ if( null == self::$a){ self::$a = new a(); } return self::$a; } }// 看到这里确实返回的是 a 的实例.但不是同一个对象.$a1 = a::getclassa();$a2 = a::getclassa();echo \$a1 的类是 .get_class($a1). , \$a2 是 .get_class($a1);if($a1 === $a2){ echo
\$a1 \$a2 指向同一对象.;}else{ echo
\$a1 \$a2 不是一个对象.;}?>
复制代码
程序运行结果为:$a1 的类是 a , $a2 是 a$a1 $a2 指向同一对象.
到此,我们写了一个最简单的单例模式 。现在,你可以尝试写一个应用单例设计模式的数据库连接类。要记住单例模式的使用效果和书写方式。
