设计模式专题(五)——工厂方法模式

设计模式专题(五)——工厂方法模式

(原创内容,转载请注明来源,谢谢)

一、概述

1、工厂方法与简单工厂模式区别

工厂方法模式与简单工厂模式不同

简单工厂模式最大优点在于工厂类类中包含了必要的逻辑判断,根据客户端的选择条件动态实例化相关的类,对于客户端来说,除去了与具体产品的依赖。但是,当需要新增、删除工厂类的功能,违反了开放-封闭原则。

工厂方法模式定义一个用于创建对象的接口,让子类决定实例化哪一个类,工厂方法使一个类的实例化延迟到其他子类。对工厂需要新增功能,只需要新增具体的工厂类,去继承工厂接口;再新增实现方法类,去继承实现接口,符合开放-封闭原则。

2、工厂方法模式存在问题

1)简单方法模式中在工厂类的逻辑判断部分,被交到客户端去实现,因此客户端处还需要进行逻辑判断的功能。

2)仅仅为了创建一个特定的ConcreteProduct对象,就不得不创建一个Creator子类。

3、类图

4、工厂方法模式结构

1)抽象工厂(Creator)角色:是工厂方法模式的核心,与应用程序无关。任何在模式中创建的对象的工厂类必须实现这个接口。

2)具体工厂(ConcreteCreator)角色:这是实现抽象工厂接口的具体工厂类,包含与应用程序密切相关的逻辑,并且受到应用程序调用以创建产品对象。

3)抽象产品(Product)角色:工厂方法模式所创建的对象的超类型,也就是产品对象的共同父类或共同拥有的接口。

4)具体产品(ConcreteProduct)角色:这个角色实现了抽象产品角色所定义的接口。某具体产品有专门的具体工厂创建,它们之间往往一一对应。

二、业务实现

1、业务场景

工厂类的设计模式最适合用于需要整合各类相似要求的业务场景,例如对数据库的操作,就可以使用工厂方法模式,现仅考虑增、删、改三个操作。

1)工厂接口

工厂接口定义具体工厂类需要的方法,在操作数据库中如连接不同表,每个表对应一个实体。

2)工厂类

工厂类继承工厂接口,其定义工厂的具体,实现对具体表产品的操作类的调用。

3)产品接口

产品接口定义产品所需要实现的功能的方法,对于每个表,有具体的方法定义。

4)产品类

产品类定义产品的具体功能实体,即对于每个表,有自己独特的增、删、改的方法。

这样的设计很灵活,且新增表格、改动表格在项目中很常见,改动后相应的增删改的操作肯定有所变动,而工厂方法模式可以很方便的对其进行改动。

另外,每个表有自己的类,因此表格的改动,可以快速找到相关客户端、服务端的调用点,进行进一步的检查。避免因某个业务调整、表格改动,导致影响其他使用该表格的业务也发生异常。

2、程序实现

<?php
//工厂方法模式,实现数据库对于每个表的增删改的定义,建立相关实体
//工厂接口
interface Factory{
       publicfunction connectDetailTable($host,$username,$password);
}
//工厂实现类——针对表table1、表table2
class table1Service implements Factory{
       publicfunction connectDetailTable($host,$username,$password){
//在已知具体某个表格的情况下,其在哪个数据库中可以确定 
              $database= 'database of table1'; 
              returnnew table1Entity(
                     newPDO('mysql:host='.$server.';dbname='.$database.'',$username,$password)
              );
       }
}
class table2Service implements Factory{
       publicfunction connectDetailTable($host,$username,$password){
              $database= 'database of table2';
              return new table2Entity(
                     newPDO('mysql:host='.$server.';dbname='.$database.'',$username,$password)
              );
       }
}
//产品接口,定义表格应该有的方法,包括新增、批量新增、修改、删除四种方法
interface Product{
       publicfunction __construct($conn);
       //createDatas=array(col1=>val1,col2=>val2...)
       publicfunction create(array $createDatas);    
//createBatchDatas=array(array(col1=>val1,col2=>val2...),array(col1=>val1...)...)
public functioncreateBatch(array $createBatchDatas);
//modifyDatas=array(array(condition1=>val1,condition2=>val2...),array(col1=>val1,col2=>val2...))
       publicfunction modify(array $modifyDatas);
//deleteDatas=array(condition1=>val1,condition2=>val2...)
       publicfunction delete(array $deleteDatas);
}
//产品实现类,实现产品接口,table1的实体
class table1Entity implements Product{
       private$conn;
       private$table;
       publicfunction __construct($conn){
              $this->conn= $conn;
              $this->table= 'table1';
       }
       publicfunction create(array $createDatas){
              //...新增逻辑,主要是针对字段的判断以及表格具体新增规则的限制,如非空、数据内容等,返回last_insert_id
       }
       publicfunction createBatch(array $createBatchDatas){
              //...批量新增逻辑,主要是针对字段的判断以及表格具体新增规则的限制,如非空、数据内容等,返回affect_rows
       }      
       publicfunction modify(array $modifyDatas){
              //...修改逻辑,主要是针对表格特有的字段进行判断,以及对特定列是否允许修改进行判断,返回affect_rows
       }
       publicfunction delete(array $deleteDatas){
              //...删除逻辑,主要是针对表格特定的删除规则,另外如果有关联其他表,还需要判断是需要删除其他表相关的数据还是不允许删除此条数据等,返回affect_rows
       }
}
//table2的实体不再描述

三、实际意义

数据库的此设计方案,在实际项目中具有重要意义。

1、规范增删改的操作,避免用户随意对表进行增删改。

2、实现对每个表的具体的判断的“定制化”,针对每个表有自己的要求去设计。

3、便于查找,降低改动的风险:其实现了工厂方法接口的灵活性,需要新增表格,只需要新增相应的service和entity;修改表格字段、删除表格,往往需要检查程序的调用点,则只需要检查相应service、entity的调用点即可。

——written by linhxx 2017.07.30

相关阅读:

设计模式专题(四)——代理模式

设计模式专题(三)——装饰模式

设计模式专题(二)——策略模式

设计模式专题(一)——面向对象的设计原则

原文发布于微信公众号 - 决胜机器学习(phpthinker)

原文发表时间:2017-07-30

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

发表于

我来说两句

0 条评论
登录 后参与评论

相关文章

来自专栏微信公众号:Java团长

Java进阶02 异常处理

程序很难做到完美,不免有各种各样的异常。比如程序本身有bug,比如程序打印时打印机没有纸了,比如内存不足。为了解决这些异常,我们需要知道异常发生的原因。对于一些...

641
来自专栏会跳舞的机器人

java并发编程的艺术笔记第一章——并发编程的挑战

单核处理器也支持多线程执行代码,CPU通过给每个线程分配CPU时间片来实现这个机制。时间片是CPU分配给各个线程的时间,因为时间片非常短,所以CPU通过不停地切...

783
来自专栏java工会

Java 动态代理及 RPC 框架介绍

所谓动态代理,指的是语言提供的一种语法,能够将对对象中不同方法的调用重定向到一个统一的处理函数中来。 python重写__getattr__函数能够做到这一点,...

551
来自专栏Vamei实验室

Java进阶02 异常处理

程序很难做到完美,不免有各种各样的异常。比如程序本身有bug,比如程序打印时打印机没有纸了,比如内存不足。为了解决这些异常,我们需要知道异常发生的原因。对于一些...

1677
来自专栏Python

python并发编程之多进程理论部分

一 并发与并行   无论是并行还是并发,在用户看来都是'同时'运行的,不管是进程还是线程,都只是一个任务而已,真是干活的是cpu,cpu来做这些任务,而一个cp...

1725
来自专栏我就是马云飞

设计模式二十四章经之责任链模式

1023
来自专栏性能与架构

Apache 的工作模式

Apache 是一个非常成熟的Web服务器,工作模式也在不断优化 现在 Apache 已经有了 3 个核心工作模式,看下他们各自的工作方式是什么样的 (1)pr...

3757
来自专栏喵了个咪的博客空间

phalapi-进阶篇2(DI依赖注入和单例模式)

#phalapi-进阶篇2(DI依赖注入和单例模式)# ? ##前言## 先在这里感谢phalapi框架创始人@dogstar,为我们提供了这样一个优秀的开源框...

3025
来自专栏python3

python3--IO模型,阻塞,非阻塞,多路复用,异步,selectors模块

结论:协程任务开启,并不一定会执行,它需要I/O(阻塞)才能执行,上面代码的time.sleep(1)模拟了I/O(阻塞)

422
来自专栏Python

IO模型

一 概念理解 在进行网络编程时,我们常常见到同步(Sync)/异步(Async),阻塞(Block)/非阻塞(Unblock)四种调用方式: 同步:   所谓同...

1725

扫描关注云+社区