首先我们要知道明确单例模式这个概念,那么什么是单例模式呢?
单例模式顾名思义,就是只有一个实例。
作为对象的创建模式, 单例模式确保某一个类只有一个实例,而且自行实例化并向整个系统提供这个实例,这个类我们称之为单例类。
单例模式的要点有3个:
1、某个类只能有一个实例;
2、它必须自行创建这个实例;
3、它必须自行向整个系统提供这个实例。
下面我们讨论下为什么要使用PHP单例模式?
多数人都是从单例模式的字面上的意思来理解它的用途, 认为这是对系统资源的节省, 可以避免重复实例化, 是一种”计划生育”. 而PHP每次执行完页面都是会从内存中清理掉所有的资源. 因而PHP中的单例实际每次运行都是需要重新实例化的, 这样就失去了单例重复实例化的意义了. 单单从这个方面来说, PHP的单例的确有点让各位失望. 但是单例仅仅只有这个功能和应用吗? 答案是否定的,我们一起来看看。
1. php的应用主要在于数据库应用, 所以一个应用中会存在大量的数据库操作, 在使用面向对象的方式开发时(废话), 如果使用单例模式, 则可以避免大量的new 操作消耗的资源。
2. 如果系统中需要有一个类来全局控制某些配置信息, 那么使用单例模式可以很方便的实现. 这个可以参看zend Framework的FrontController部分。
3. 在一次页面请求中, 便于进行调试, 因为所有的代码(例如数据库操作类db)都集中在一个类中, 我们可以在类中设置钩子, 输出日志,从而避免到处var_dump, echo。
多例:
- 顾名思义就是多个实例
- 例如-实例可以被new为多个相同的实例,单例的话只能允许有一个实例
- 那么简单来说多例就是拥有多个单例
- 以连接数据库为例,他拥有更高的维护性
举个栗子
<?php
/**
* 多例(使用final修饰是为了类禁止被继承)
* 读写分离
* 只能创建两个示例
*/
final class db
{
/**
* 保存全局实例
* @var array[PDO]
*/
private static $instance;
/**
* 禁止外界实例化对象.
*/
private function __construct(int $type)
{
echo PHP_EOL . "我被连接1次$type";
return new PDO("mysql:host=127.0.0.1;dbname=fast;port=3306", 'root', 'e002d8b739ea9407');
}
/**
* 禁止外界克隆对象
*/
public function __clone()
{
}
/**
* 禁止外界序列化对象
*/
public function __wakeup()
{
}
/**
* Db_Query $type=0
* Db_Write $type=1
*/
public static function getInstance(int $type)
{
if (!(self::$instance[$type] instanceof self)) {
self::$instance[$type] = new self($type);
} else {
echo PHP_EOL . "我连接好了$type";
}
return self::$instance[$type];
}
}
/**
* 单例模式(整个类叫做实例,只能被new一次叫做单例模式)
*/
class class1
{
/**
* 保存全局实例.
* @var class1
*/
private static $instance;
public $a = 1;
/**
* 私有化构造函数,防止外界实例化对象.
*/
private function __construct()
{
echo "我被new了";
// 只会被new一次,可以做数据库链接操作
}
/**
* 私有化克隆函数,防止外界克隆对象.
*/
private function __clone()
{
}
/**
* 单例访问统一入口.
* @return \class1
*/
public static function getInstance(): class1
{
if (!(self::$instance instanceof self)) {
self::$instance = new self();
}
return self::$instance;
}
/**
* 测试哦
*/
public function test(): void
{
echo $this->a;
$this->a++;
// 输出多例
db::getInstance(0);
db::getInstance(1);
}
}
for ($i = 0; $i < 10; $i++) {
class1::getInstance()->test();
echo PHP_EOL;
}
单例输出:
我被new了1
我被连接1次0
我被连接1次1
2
我连接好了0
我连接好了1
3
我连接好了0
我连接好了1
4
我连接好了0
我连接好了1
5
我连接好了0
我连接好了1
6
我连接好了0
我连接好了1
7
我连接好了0
我连接好了1
8
我连接好了0
我连接好了1
9
我连接好了0
我连接好了1
10
我连接好了0
我连接好了1