PHP之设计模式

https://blog.csdn.net/self_realian/article/details/78228733

掌握PHP各类设计模式,具备设计纯面向对象框架和系统能力是非常有必要的。给大家一个小建议吧,学习设计模式的时候,使用phpStorm工具,因为设计模式是纯面向对象的,有很多的类,phpStorm 这个IDE工具有很多的快捷方式能帮助你快速找到对应的类。那么废话少数说,下面进入正题吧。

一、面向对象的高级特性

1,命名空间

最早的PHP是没有命名空间的,所有的代码都是在同一个命名空间下,这样就不能重载相同的类或函数,随着我们项目的工程越来越大,文件越来越多,这个时候产生命名冲突就会越来越多,代码也会越来越复杂。到PHP5.3的时候,出现了命名空间。规定所有命名空间放在文件的开头位置(关于命名空间的具体语法及使用这里就不说了,大家自行去学习。本篇博客主要说的是设计模式)

2,类自动载入

最早的时候我们载入类都是通过include或者require来实现,当PHP项目变大之后,载入的文件越来越多,那如果一个文件要依赖几十个类文件,那就要在这个文件的前边写几十个require,这样会给开发带来不便。通过自动载入就能很方便的载入我们需要的类(自动载入的知识也需要大家自己去学习)

前边我们提了命名空间和类自动载入,那么现在我们就用这两个知识来搭一个PHP面向对象的基础框架。首先我们来说一下PSR-0这个规范(PHP面向对象编码规范)。这个规范有三项约定

1,PHP的命名空间必须与绝对路径一致

2,类名的首字母必须大写

3,除了入口文件之外,其它“.php”必须只有一个类,不能有其它的可执行的代码

开发一个符合PSR-0规范的基础框架

1,全部使用命名空间

2,所有PHP文件必须自动载入,不能有include/require

3,单一入口文件

PSR-0规范框架代码

首先我给我的目录先截一下图

PHP之设计模式

index.php

<?php

//入口文件

define('BASEDIR', __DIR__);//整个框架的根目录

include BASEDIR.'/Shulv/Loader.php';

spl_autoload_register('\\Shulv\\Loader::autoload');

Shulv\Object::test();

App\Controller\Home\Index::test();

Object.php

<?php

namespace Shulv;

class Object

{

static function test(){

echo "test()"."<br />";

}

}

//不能再有任何其它可执行的代码

Loader.php

<?php

namespace Shulv;

class Loader{

static function autoload($class){

require BASEDIR.'/'.str_replace('\\', '/', $class).'.php';

}

}

Index.php

<?php

namespace App\Controller\Home;

class Index{

static function test(){

echo __METHOD__;

}

}

这里主要学的是思想,框架代码是很简单的。

3,SPL标准库简介(PHP标准库)

SPL中提供了很多常用数据结构的一些封装,具体的可以去php的官方网站(www.php.net/spl)来查看有关SPL的一些东西。

//比如栈结构(先进后出)

$stack = new SplStack();//栈主要有两种接口,就是进(push())和出(pop())

$stack->push("data1\n");//入栈

$stack->push("data2\n");//入栈

echo $stack->pop();//出栈

echo $stack->pop();//出栈

输出结果:

data2

data1

//队列结构

$queue = new SplQueue();//队列是先进先出

$queue->enqueue("data1\n");//入队列

$queue->enqueue("data2\n");//入队列

echo $queue->dequeue();//出队列

echo $queue->dequeue();//出队列

输出结果:

data1

data2

//堆结构

$heap = new SplMinHeap();

$heap->insert("data1\n");//将数据插入到堆

$heap->insert("data2\n");//将数据插入到堆

echo $heap->extract();//从堆中取出数据

echo $heap->extract();

运行结果

data1

data2

本篇博客主要是讲集中基础的设计模式:

1,工厂模式

2,单例模式

3,注册模式

一,工厂模式

我们可以设计出来一个工厂(其实就是类),该工厂的作用就是为我们生产各种对象。这种工厂通常只要指定类名,就可以据此获取

<?php

namespace Shulv;

class Factory{

//Instance表示“实例”,也就是指“对象”

static function getInstance($className){//将要实例化的类名传递过来

if(file_exists('./'.$className.'.class.php')){

$obj1 = new $className();

return $obj1;

}else{

return 'not exist';

}

}

}

二,单例模式

对于某些类,在使用它的时候,从头到尾(从程序的开始到结束),都只需要一个对象,就可以完成所有的人物

* 单例:

* 某个类只允许创建出一个对象,即使去创建多次,也只能得到一个对象(就比如创建一个连接数据库的对象,前前后后只要创建一个就行)

//实现一个单例类:其只能获得一个对象

class B{

private static $instance;//定义成私有,那么它只能在本类中使用。定义成静态就能通过类名来访问了

private function __construct(){}//定义成私有,也就是不让别人在类外new 出对象,这样就只能通过getNew()获取对象

public static function getNew(){

//如果本类中的$instance还没有数据

if(!isset(B::$instance)){

B::$instance = new self;

}

return B::$instance;

}

private function __clone()

{

}

}

$o1 = B::getNew();

$o2 = B::getNew();

//我们会发现这创建的还是一个类(资源标识符相同)

//var_dump($o1,$o2);

//下面我们再看一下一个特殊情况

$o3 = clone $o1;//克隆对象(当我把那个__clone()方法定义成私有,这句就会报错)

var_dump($o1,$o2,$o3);//这个时候我们会发现$o3是一个新对象了   这个时候又不符合单例了(注意:这个clone算运算符)

/*

*本来一个类中有一个隐藏的public function __clone()方法,那我们如果想实现单例,我们给它定义成私有就行,不让他在类外使用

*/

三,注册模式

全局共享和交换对象

<?php

namespace Shulv;

/*

注册模式就是用来将一些对象注册到我们的全局事物上边,那就可以在任意位置访问

*/

class Register{

protected static $objects;

function set($alias, $object){//表示将一个对象注册到全局的注册树上

self::$objects[$alias] = $object;//第一个参数表示取的别名

}

function _unset($alias){//注意:这个_unset()不是魔术方法,魔术方法那个是__unset()(两个下划线)

//表示从树上移除

unset(self::$objects[$alias]);

}

function get($name){

return self::$objects[$alias];

}

}

//当我们在工厂模式中获得对象之后,再将获得的对象映射到全局树上,那么工厂模式就只用构造一次

Register::set('A',$a);//A是别名(加在工厂模式中)

上一篇:入门ROS教程与视频汇总(kinetic)


下一篇:C#中另类自定义公式计算 字符串转换为计算公式,并得出计算结果