Home >Backend Development >PHP Tutorial >看看这个类的实例化方式能否实现“单例模式”?
比如我有一个类,类名是A(这个类本身不是实现单例模式的,看后面实例化这个类的方式):
<code>class A(){ public $str = '这是属性'; public $str2 = '属性'; private $str3 = '私有属性'; public function __construct($a,$b,$c){ //构造方法 } public function func1(){ //类的方法 } public function func2(){ //类的方法 } }</code>
还有一个全局函数:
<code>function getObj(){ require PATH.'a.class.php';//引入上面定义的class A的文件 static $instance;//定义静态变量 if($instance){ return $instance; } $instance = new A('1','2','3'); return $instance; }</code>
然后我要在其他地方调用类A的实例,并且可能需要一次请求调用多次,通过getObj方法来得到这个类的实例,因为是返回的静态变量,其实在第一次创建这个类的实例后,后面调用都是直接用的这个静态变量的实例,并没有重新new一个新的实例,这种方式算不算单例模式??如果不算,主要是哪里不同会有什么隐患??十分感谢!
比如我有一个类,类名是A(这个类本身不是实现单例模式的,看后面实例化这个类的方式):
<code>class A(){ public $str = '这是属性'; public $str2 = '属性'; private $str3 = '私有属性'; public function __construct($a,$b,$c){ //构造方法 } public function func1(){ //类的方法 } public function func2(){ //类的方法 } }</code>
还有一个全局函数:
<code>function getObj(){ require PATH.'a.class.php';//引入上面定义的class A的文件 static $instance;//定义静态变量 if($instance){ return $instance; } $instance = new A('1','2','3'); return $instance; }</code>
然后我要在其他地方调用类A的实例,并且可能需要一次请求调用多次,通过getObj方法来得到这个类的实例,因为是返回的静态变量,其实在第一次创建这个类的实例后,后面调用都是直接用的这个静态变量的实例,并没有重新new一个新的实例,这种方式算不算单例模式??如果不算,主要是哪里不同会有什么隐患??十分感谢!
接下去就要是PHP7的时代了,你的代码还停留在看上去像是5.x早期时代的PHP代码,本身代码应该可以跑通,你的使用方式从宽泛的定义上来讲,勉强可以算单例,但是整体感觉非常奇怪。并且如别的回答提出的,如果别人不知道getObj函数的话,很有可能就自己new了,你的获取对象的函数无法对别人起到约束作用
既然你的构造函数依赖三个参数,这是我对于这份代码的建议实现
<code>class A(){ public $str = '这是属性'; public $str2 = '属性'; private $str3 = '私有属性'; private static $pool = []; public static function getInstance($a, $b, $c) { /** * 这里假设一种情况是需要这三个参数这个对象才唯一 * 并且这三个参数都是可以转换为字符串类型的 *(也就是说不能是stream,array,或者没有__toString()实现的object) */ $key = implode('_', func_get_args()); if (!isset(self::$pool[$key])) { /** * 关于这里使用static关键字详情请看后期静态绑定 * http://php.net/manual/zh/language.oop5.late-static-bindings.php * 如果你的PHP还在用5.3以前的版本,请使用 * self::$pool[$key] = new self($a, $b, $c); */ self::$pool[$key] = new static($a, $b, $c); } return self::$pool[$key]; } protected function __construct($a, $b, $c) { //构造方法,没有使用private是考虑如果有子类继承的情况 } public function func1() { //类的方法 } public function func2() { //类的方法 } }</code>
PS 另外建议题主去了解一下PSR规范和autoload
会报错的,而且这种写法很糟糕
其实不算。原因如下:
1.假设是多线程的环境,多个线程同时进入会不会同时产生多个对象呢,所以需要你进行冗余处理。我不知道php是怎么做的,但是Java可以通过sync关键字来搞定同步。
2.你构造函数是公有的,假设别人在你的代码上二次开发,但他不知道你的这个getobj。他就会自己new一个。这样也会有问题。