上一篇文章讲了一下什么是单例模式并用一个简单的例子说明了一下,但是在实际开发中,可能会有另外一种比较特殊的情况出现,比如数据库连接对象,在一些大型的应用中,你可能需要连接多台数据库,这里多个数据库公用一个数据库连接对象,可能会产生一些意想不到的问题,比如连接的分配、获取insert_id(插入的id)last_error(最后的错误)等.
这个问题也比较好解决,就是把我们的 $instance 变成一个关联数组,通过给 getInstance 方法传入不同的参数获取不同的"单例对象"(引号的含义是:严格来说类可能被new多次,但是这个new也是在我们的控制之内的,而不是在类外部):
class MysqlServer{
//注意,变成复数, 只是为了标识
private static $instances = array();
//业务变量,保持当前实例的mysqli对象
private $conn;
//显著特征:私有的构造方法,避免在类外部被实例化
private function __construct($host, $username, $password, $dbname, $port){
$this->conn = new mysqli($host, $username, $password, $dbname, $port);
}
//类唯一实例的全局访问点
public static function getInstance($host='localhost', $username='root', $password='123456', $dbname='mydb', $port='3306'){
$key = "{$host}:{$port}:{$username}:{$dbname}";
if (empty(self::$instances[$key])){
//这里也可以用 new self(); 的方式
$class = __CLASS__;
self::$instances[$key] = new $class($host, $username, $password, $dbname, $port);
}
return self::$instances[$key];
}
//重载__clone方法,不允许对象实例被克隆
public function __clone(){
throw new Exception("Singleton Class Can Not Be Cloned");
}
//查询业务方法,后面省略其它业务方法
public function query($sql){
return $this->conn->query($sql);
}
//尽早释放资源
public function __destruct(){
$this->conn->close();
}
}