工厂设计模式

ops/2024/9/25 10:34:33/

工厂设计模式是一种创建型设计模式,用于创建对象的过程中封装对象的创建逻辑。

它提供了一种创建对象的最佳方式,不需要在每个使用该对象的地方直接实例化对象。工厂模式通过定义一个公共的接口来创建对象,然后由子类来实现这个接口以提供具体的对象实例化逻辑 

核心本质:

  1. 实例化对象不使用new,用工厂方法代替
  2. 选择实现类,创建对象统一管理和控制,从而将调用者跟我们的实现类解耦

工厂设计模式主要有三种形式:简单工厂模式、工厂方法模式和抽象工厂模式。

  1. 简单工厂模式(Simple Factory Pattern):简单工厂模式通过一个工厂类来负责创建所有的产品类的实例,客户端只需要提供工厂类所需的参数,即可获取所需的产品实例。

  2. 工厂方法模式(Factory Method Pattern):工厂方法模式将工厂抽象成一个接口或抽象类,每个具体的工厂类负责创建一种产品,客户端通过调用工厂方法来获取产品实例。

  3. 抽象工厂模式(Abstract Factory Pattern):抽象工厂模式提供一个接口,用于创建一系列相关或依赖对象,而不需要指定具体的类。客户端通过调用工厂接口来获取不同产品的实例。

工厂模式的优点包括:封装了对象的创建逻辑,降低了客户端和具体产品类之间的耦合度,提高了代码的可维护性和扩展性。缺点是如果产品类的种类过多,会导致工厂类代码变得复杂,维护困难。

创建一个接口,定义了我们要实现的基本功能

java">public interface Car {void name();
}

创建这个接口的实现类

java">public class TSLa implements Car {@Overridepublic void name() {System.out.println("特斯拉");}
}
java">public class WuLing implements Car {@Overridepublic void name() {System.out.println("五菱宏光");}
}

 简单工厂模式

创建一个CarFactory类,在这个类中提供创建类的实例的静态方法,所以简单工厂也称为静态工厂。

java">public class CarFactory {//方式1//简单工厂模式/静态工厂模式//缺点:如果需要增加新的产品,需要修改代码,不符合开闭原则public static Car getCar(String name){if(name.equals("特斯拉")){return new TSLa();}else if(name.equals("五菱")){return new WuLing();}else{return null;}}//方式2public static Car getTSCar(){return new TSLa();}public static Car getWLCar(){return new WuLing();}
}

消费者在创建对应的对象的时候调用工厂类。提供对应的参数即可

java">public class test {public static void main(String[] args) {//传统的方式创建对象Car wuli=new WuLing();Car TS=new TSLa();wuli.name();TS.name();//使用简单工厂创建对象Car car = CarFactory.getCar("五菱");Car car2 = CarFactory.getCar("特斯拉");car.name();car2.name();}
}

简单工厂模式的缺点:如果我们要新增加一个类,我们就需要对CarFactory类中的代码进行修改。这违反了开闭原则。

工厂方法模式

就是给每一种汽车类都封装对应的工厂

创建一个CarFactory的接口

java">public interface CarFactory {Car getCar();
}

不同的类创造不同的工厂对象来继承这个CarFactory接口

java">public class TSLFactory implements CarFactory{@Overridepublic Car getCar() {return new TSLa();}
}
java">public class WLFactory implements CarFactory{@Overridepublic Car getCar() {return new WuLing();}
}

然后在消费者中通过不同的工厂得到对应的对象

java">public class test {public static void main(String[] args) {Car car1 = new TSLFactory().getCar();Car car2 = new WLFactory().getCar();car1.name();car2.name();}
}

如果我们需要添加一个其他的汽车类,就需要创建一个对应的汽车工厂类来创建这个汽车类的对象

如果我们的汽车类特别多,我们创建的java文件数量就会很多。

抽象工厂模式

定义:抽象工厂提供了一个创建一系列相关或者相互依赖对象的接口,无虚指定它们具体的类

适用场景:

  • 客户端(应用层)不依赖于产品类实例如何被创建、实现等细节
  • 强调一系列相关的产品对象(属于统一产品族)一起使用创建对象需要大量的重复代码
  • 提供一个产品类的库,所有的产品以同样的接口出现,从而使得客户端不依赖于具体的实现

优点:

  • 具体产品在应用层的代码隔离,无需关心创建的细节
  • 将一系列的产品统一到一起创建

缺点:

  • 规定了所有可能被创建的产品集合,产品族中扩展性的产品困难
  • 增加了系统的抽象性和理解难度

举例:现在有一个大的抽象工厂:IProductFactory,用来创建手机和路由器,手机、路由器的种类有小米和华为。

java">//抽象产品工厂
public interface IProductFactory {IPhoneProduct getPhoneProduct();RouterProduct getRouterProduct();
}
java">//华为生产工厂
public class HuaWeiFactory implements IProductFactory{@Overridepublic IPhoneProduct getPhoneProduct() {return new HuaWeiPhone();}@Overridepublic RouterProduct getRouterProduct() {return new HuaWeiRouter();}
}
java">//小米生产工厂
public class XiaoMiFactory implements IProductFactory{@Overridepublic IPhoneProduct getPhoneProduct() {return new XiaoMiPhone();}@Overridepublic RouterProduct getRouterProduct() {return new XiaoMiRouter();}
}
java">//手机产品接口
public interface IPhoneProduct {void start();void shutdown();void callup();void sendSMS();
java">//路由器产品接口
public interface RouterProduct {void start();void shutdown();void openWifi();void setting();
}
java">//华为手机实现类
public class HuaWeiPhone implements IPhoneProduct{@Overridepublic void start() {System.out.println("华为手机开机");}@Overridepublic void shutdown() {System.out.println("华为手机关机");}@Overridepublic void callup() {System.out.println("华为手机打电话");}@Overridepublic void sendSMS() {System.out.println("华为手机发短信");}
}
java">//华为路由器实现类
public class HuaWeiRouter implements RouterProduct{@Overridepublic void start() {System.out.println("华为路由器开机");}@Overridepublic void shutdown() {System.out.println("华为路由器关机");}@Overridepublic void openWifi() {System.out.println("华为路由器打开wifi");}@Overridepublic void setting() {System.out.println("华为路由器设置");}
}
java">//小米手机实现类
public class XiaoMiPhone implements IPhoneProduct{@Overridepublic void start() {System.out.println("小米手机开机");}@Overridepublic void shutdown() {System.out.println("小米手机关机");}@Overridepublic void callup() {System.out.println("小米手机打电话");}@Overridepublic void sendSMS() {System.out.println("小米手机发短信");}
}
java">//小米路由器实现类
public class XiaoMiRouter implements RouterProduct{@Overridepublic void start() {System.out.println("小米路由器开机");}@Overridepublic void shutdown() {System.out.println("小米路由器关机");}@Overridepublic void openWifi() {System.out.println("小米路由器打开WIFI");}@Overridepublic void setting() {System.out.println("小米路由器设置");}
}

客户端使用

java">public class client {public static void main(String[] args) {System.out.println("========小米=======");XiaoMiFactory xiaoMiFactory = new XiaoMiFactory();IPhoneProduct xiaomiPhone = xiaoMiFactory.getPhoneProduct();xiaomiPhone.start();xiaomiPhone.sendSMS();RouterProduct xiaomiRouter = xiaoMiFactory.getRouterProduct();xiaomiRouter.start();xiaomiRouter.openWifi();System.out.println("========华为=======");HuaWeiFactory huaWeiFactory = new HuaWeiFactory();IPhoneProduct huaweiPhone = huaWeiFactory.getPhoneProduct();huaweiPhone.start();huaweiPhone.sendSMS();RouterProduct huaweiRouter = huaWeiFactory.getRouterProduct();huaweiRouter.start();huaweiRouter.openWifi();}
}

关系图

总结

简单工厂模式(静态工厂模式)

  • 虽然不符合设计原则,但是使用方便,实际使用最多

工厂方法模式

  • 不修改已有类的前提下,通过增加新的工厂类实现扩展

抽象工厂模式

  • 不可以增加产品,但可以增加产品族

应用场景

  • JDK中Calendar的getInstance方法
  • JDBC中的Connection对象的获取
  • Spring中IOC容器创建管理的bean方法
  • 反射中Class对象的newInstance方法


http://www.ppmy.cn/ops/8718.html

相关文章

网络安全威胁激增,数据使用需更加谨慎

网络犯罪分子是指那些故意从事网络恶意活动的个人或团体。通过黑客攻击、网络钓鱼等手段意图破坏我们的系统、网络或数据。他们的动机多样,包括经济利益、政治意识形态、怨恨、复仇等等。 近年来,网络安全威胁呈现爆炸式增长。IBM的威胁情报报告将网络钓…

mapbox控制3D模型旋转

贴个群号 WebGIS学习交流群461555818,欢迎大家 效果 原理与源码 获取角度,然后一直更改角度,角度到达180度后赋值成-180度,然后转到开始获取的角度的角度的时候就停止旋转 function rotateModel(layerID){let bearing map.get…

Linux的图形资源及指令

一、火车 1.切换到超级用户 su 2.下载资源 yum install -y sl 3.输入指令 sl,得到火车图形 如果没有得到该图形,就将2处改为yum install -y epel-release。 二、Linux的logo 1.在超级用户模式下下载资源 yum install -y linux_logo 2.输…

Java初学日记 十三 (GUI)

GUI编程 概述 GUI(Graphical Uers Interface)全称图形用户界面 swing指javax.swing包,该包中包含实现界面的类,这些类都可称为组件 组件可分为两大类: 容器组件 窗口 import javax.swing.*; ​ public class LoginFrame extends JFram…

Android活动之Intent

Intent Intent是Android程序中各组件之间进行交互的一种重要方式,它不仅可以指明当前组件想要执行的动作,还可以在不同组件之间传递数据。Intent一般可被用于启动活动、启动服务以及发送广播等场景, 显示intent 第一个参数Context要求提供…

RestClient操作Elasticsearch(Java)

Es官方提供了各种不用语言的客户端,用来操作Es,这些客户端的本质就是组装DSL语句,通过http请求发送给Es,从而简化操作 es基础篇不熟悉参考一下博客:ElasticSearch入门篇-CSDN博客文章浏览阅读445次,点赞7次…

宝塔面板使用docker+nginx+gunicorn部署Django项目实战教程

第一步:创建Django项目 使用pip install django安装创建django项目的依赖在电脑某个根目录下执行django-admin startproject app创建一个名为app的Django项目。目录结构如下: ├── app │ ├── init.py │ ├── asgi.py │ ├── settings.py │ ├── url…

从零开始写一个RTSP服务器(三)RTP传输H.264

目录 一、RTP封装1.1 RTP数据结构1.2 源码 二、H.264的RTP打包2.1 H.264格式2.2 H.264的RTP打包方式2.3 H.264 RTP包的时间戳计算2.4 源码 三、H.264 RTP打包的sdp描述四、测试 本篇文章目标,使用vlc打开sdp文件后,可以观看到视频数据 一、RTP封装 1.1 …