2018-04-08php实战设计模式
2018-04-08 本文已影响0人
扎扎瑜
一、单例模式
单例模式是最经典的设计模式之一,到底什么是单例?单例模式适用场景是什么?单例模式如何设计?php中单例模式具体如何实现?
单例模式:也叫原子模式(职责模式),是一种常见的设计模式。在应用这个模式时,单例对象的类只能有一个实例。
通俗易懂的说:某一个功能只能实例化一个对象。
单例模式适用场景
实际项目中像数据查询、日志输出、全局回调、统一校验等模块。这些模块功能单一,但需要多次访问,如果能够全局唯一,多次复用会大大提升性能。
单例模式的好处:
1.减少频繁创建,节省了cpu;
2.静态对象公用,节省了内存;
3.功能解耦,易于维护;
如何设计单例模式
单例模式的核心:实例一次生成,全局唯一,多次调用
,因此单例模式有三要素:
1.私有化构造函数,私有化clone,也就不能new,不能clone(唯一)
2.有一个静态变量,用来保存当前类。(唯一如何保存)
3.提供一个公共的访问入口(可以访问)
php实现(例子一)
<?php
class Singleton_Mode
{
# 私有化构造函数
private function __contruct()
{
}
# 私有化clone
private function __clone()
{
}
# 保存实例的静态对象
public static $singletonStatus;
/**
* 声明静态调用方法
* 目的:保证该方法的调用全局唯一
* @return Singleton_Mode
*/
public static function getSingleton()
{
if(!self::$singletonStatus)
{
self::$singletonStatus =new self;
}
return self::$singletonStatus;
}
# 调用单例模式
public function singletonFun()
{
echo "dan li";
}
}
# 这么写是错误的 因为构造函数被申明为private 错误start
# 实例化对象
# 错误start
❌$singletonStatus = new Singleton_Mode();
❌$singletonStatus->singletonFun();
# 错误end
# 现在是该类的单例对象
$test1 = Singleton_Mode::getSingleton();
$test2 = Singleton_Mode::getSingleton();
var_dump($test1 == $test2);
#测试看看是否可以clone
// $test_clone = clone $test;
php实现(例子二)
<?php
/**
* 创建型模式
*
* php单例模式
*
* @example 运行 php test.php
*/
// 注册自加载
spl_autoload_register('autoload');
function autoload($class)
{
require dirname($_SERVER['SCRIPT_FILENAME']) . '//..//' . str_replace('\\', '/', $class) . '.php';
}
/************************************* test *************************************/
use singleton\Singleton;
// 获取单例
$instance = Singleton::getInstance();
$instance->test();
// clone对象试试
$instanceClone = clone $instance;
二、工厂模式(test)
<?php
/**
* 创建型模式
* 工厂方法模式和抽象工厂模式的核心区别
* 工厂方法模式利用继承,抽象工厂模式利用组合
* 工厂方法模式产生一个对象,抽象工厂模式产生一族对象
* 工厂方法模式利用子类创造对象,抽象工厂模式利用接口的实现创造对象
* 工厂方法模式可以退化为简单工厂模式(非23中GOF)
*
* php工厂模式
* @example 运行 php test.php
*/
// 注册自加载
spl_autoload_register('autoload');
function autoload($class)
{
require dirname($_SERVER['SCRIPT_FILENAME']) . '//..//' . str_replace('\\', '/', $class) . '.php';
}
/************************************* test *************************************/
use factory\Farm;
use factory\Zoo;
use factory\SampleFactory;
// 初始化一个工厂
$farm = new Farm();
// 生产一只鸡
$farm->produce('chicken');
// 生产一只猪
$farm->produce('pig');
// 初始化一个动物园工厂
$zoo = new Zoo();
$zoo->produce("chicken");
$zoo->produce("pig");
// 工厂方法模式退化为简单工厂模式
SampleFactory::produce("chicken");
SampleFactory::produce("pig");
三、抽象工厂模式(test)
<?php
/**
* 创建型模式
*
* php抽象工厂模式
*
* 说说我理解的工厂模式和抽象工厂模式的区别:
* 工厂就是一个独立公司,负责生产对象;
* 抽象工厂就是集团,负责生产子公司(工厂);
*
* @example 运行 php test.php
*/
// 注册自加载
spl_autoload_register('autoload');
function autoload($class)
{
require dirname($_SERVER['SCRIPT_FILENAME']) . '//..//' . str_replace('\\', '/', $class) . '.php';
}
/************************************* test *************************************/
use factoryAbstract\AnimalFactory;
use factoryAbstract\PlantFactory;
// 初始化一个动物生产线, 包含了一族产品
$animal = new AnimalFactory();
// 初始化一个植物生产线, 包含了一族产品
$plant = new PlantFactory();
// 模拟调用, 抽象工厂模式核心是面向接口编程
function call(factoryAbstract\Factory $factory) {
$earn = function(factoryAbstract\Income $income) {
$income->money();
};
$earn($factory->createFarm());
$earn($factory->createZoo());
}
call($animal);
call($plant);
四、原型模式(test)
<?php
/**
* 创建型模式
*
* php原型模式
* 用于创建对象成本过高时
*
* @example 运行 php test.php
*/
// 注册自加载
spl_autoload_register('autoload');
function autoload($class)
{
require dirname($_SERVER['SCRIPT_FILENAME']) . '//..//' . str_replace('\\', '/', $class) . '.php';
}
/************************************* test *************************************/
use prototype\Prototype;
// 创建一个原型对象
$prototype = new Prototype();
// 获取一个原型的clone
$prototypeCloneOne = $prototype->getPrototype();
$prototypeCloneOne->_name = 'one';
$prototypeCloneOne->getName();
// 获取一个原型的clone
$prototypeCloneTwo = $prototype->getPrototype();
$prototypeCloneTwo->_name = 'two';
$prototypeCloneTwo->getName();
// 再次获取$prototypeCloneOne的名称
$prototypeCloneOne->getName();