设计模式-简单工厂Coding+jdk源码解析

  

  前面的软件设计七大原则,目前只有理论这块,因为最近参与项目重构,暂时没有时间把Coding的代码按照设计思路一点点写出来。本周周末会花时间整理出来,然后现在想的是白天更新一点并发编程,晚上回家学习设计模式。非科班出身,脑子也比较笨。博文都是自己根据学习的时候所想的思路,如果能有帮到各位的地方,那十分荣幸。如果有欠缺之处,希望能在评论中指出一起进步。好啦,开始正文了。

本套设计模式的博文,包含各种设计模式的定义、类型、适用场景及优缺点分析。并通过Coding去实际加深理论理解。

简单工厂:

  该模式本身使用于业务较为简单的情况,一般用于小项目或具体产品很少扩展的情况(这样工厂类不用经常修改)。

  定义可以归为一句话:由一个工厂对象决定创建出哪一种产品类的实例,它属于创建型模式,但并不属于GOF23种设计模式。

  组成角色:

    工厂类角色:本模式的核心,包含一定的商业逻辑和判断逻辑。根据逻辑不同,产生具体的工厂产品。

    抽象产品角色:一般是具体产品继承的父类或实现的接口。

    具体产品角色:工厂类所创建的对象就是此角色的实体,在java中由一个具体类实现。

  该模式适用场景:

    1、工厂类负责创建的对象比较少。

    2、客户端(应用层)只知道传入工厂类的参数。对于如何创建对象(逻辑)不关心。

  简单工厂-优点:

    只需要传入一个正确的参数,就可以获取你所需要的对象而无须知道其他创建细节

  简单工厂-缺点:

    工厂类的职责相对过重,增加新的产品。需要修改工厂类的判断逻辑,违背开闭原则

  实际上这都只是为了方便理解该模式做的铺垫。接下来通过Coding+小案例理解该模式。

  案例场景:

    对于学习编程现在有很多视频教学,我们现在有一个需求就是实现不同的课程录制不同的视频。那么我们对于案例来创建对应的角色。

    首先录制视频是一个公共的行为,那么我们将视频作为抽象的产品角色,创建一个视频抽象类Video,并编写一个produce的方法。

    

    对于不同的课程录制不同视频,我们可以创建具体的产品角色类。此处是JavaVideo和PythonVideo并继承Video,实现各自的功能。(我的截图真心很丑,大家凑合着看吧)

    

    此时我们通过应用层类Test来实现该需求。

    

    我们可以通过new 不同的对象,实现不同的视频录制。但这样会造成应用层要实现什么功能,就要去new对应的实现类。应用层过度依赖对应的类

    通过简单工厂的模式,我们解除应用层对这些实现类的依赖关系

    创建工厂类角色VideoFactory,通过工厂类去依赖对应的实现类。

    

    工厂类包含业务的判断逻辑。根据逻辑不同,产生具体的工厂产品。而此时的Test变为

    

    这里就体现了简单工厂模式的适用场景,应用层Test不需要关心如何创建对象,只传入工厂类的参数。这样正是简单工厂模式的优点。

    到这里这个案例好像就完成了,但是仔细想想简单工厂模式的缺点。当我们需要实现的视频种类越多时,我的工厂类新增逻辑判断越多,这就违背了开闭原则。对于基类不应该频繁修改。

    那么我们再来优化一下这个代码:

    工厂类:

    

    应用层

    

    这里我们利用的Java的反射机制,对于工厂类我们通过应用层传入的实现类的类对象JavaVideo.class,获取传入参数的类名并通过Class.forName方法加载这个类,通过newInstance方法去获取对象。实现了不同的业务实现同时不需要在工厂类中去新增逻辑判断了。这样就解决了简单工厂模式的缺点。

    那么案例的Coding就到这里了,今天太晚了,关于jdk源码解析,那就明晚再写吧。

  

原文地址:https://www.cnblogs.com/zhangbLearn/p/9545485.html

时间: 2024-11-12 18:20:18

设计模式-简单工厂Coding+jdk源码解析的相关文章

设计模式课程 设计模式精讲 8-11 单例模式源码解析(jdk+spring+mybaties)

1 源码解析 1.1 单例解析1 1.2 单例解析2(容器单例) 1.3 单例解析3 1.4 单例解析4 1 源码解析 1.1 单例解析1 java.lang.Runtime /** * 饿汉式加载,初始化的时候,就已经new出了对象 */ private static Runtime currentRuntime = new Runtime(); /** * Returns the runtime object associated with the current Java applicat

JDK 源码解析 —— ConcurrentHashMap

零. 概述 ConcurrentHashMap 是将锁的范围细化来实现高效并发的. 基本策略是将数据结构分为一个一个 Segment(每一个都是一个并发可读的 hash table, 即分段锁)作为一个并发单元. 为了减少开销, 除了一处 Segment 是在构造器初始化的, 其他都延迟初始化(详见 ensureSegment). 并使用 volatile 关键字来保证 Segment 延迟初始化的可见性问题. HashMap 不是线程安全的, 故多线程情况下会出现 infinit loop.

JDK 源码解析 —— Executors ExecutorService ThreadPoolExecutor 线程池

零. 简介 Executors 是 Executor.ExecutorService.ThreadFactory.Callable 类的工厂和工具方法. 一. 源码解析 创建一个固定大小的线程池:通过重用共享无界队列里的线程来减少线程创建的开销.当所有的线程都在执行任务,新增的任务将会在队列中等待,直到一个线程空闲.由于在执行前失败导致的线程中断,如果需要继续执行接下去的任务,新的线程会取代它执行.线程池中的线程会一直存在,除非明确地 shutdown 掉. public static Exec

设计模式课程 设计模式精讲 12-3 适配器模式源码解析

1 源码解析 1.1 源码解析1(在jdk中的应用) 1.2 源码解析2(Spring中的通知管理) 1.3 源码解析3(SpringMVC中的应用) 1 源码解析 1.1 源码解析1(在jdk中的应用) xmlAdapter(此类是用于适配xml的一个类,是处理xml序列化和反序列化的一个类) public abstract class XmlAdapter<ValueType,BoundType> { /** * Do-nothing constructor for the derived

设计模式课程 设计模式精讲 21-3 观察者模式源码解析

1 源码解析 1.1 源码解析1(jdk中的应用) 1.2 源码解析2(Spring中的应用) 1.3 源码解析为何算出666?   1 源码解析 1.1 源码解析1(jdk中的应用) java.util.regex.Pattern (java正则表达式通过java.util.regex包下的Pattern类与Matcher类实现) 1.2 源码解析2(Spring中的应用) 测试类:(通过el表达式计算出结果)(以下代码中红色粗体为跳转顺序) package com.geely.design.

JDK 源码解析 —— HashSet

零. 简介 这个类实现了 Set 接口,内部是由 HashMap 实例辅助实现的.它不保证元素的顺序,数据允许为 null. 假如 hash 方法将数据分散地比较合理,比如一个 bucket 一个数据,那么 add.remove.contains.size 性能开销是常数时间. 这个类非线程安全的,如果多线程并发访问,并且至少一个线程在做修改操作,那么必须在外部做好同步处理.例如使用:Set s = Collections.synchronizedSet(new HashSet(...)); 一

JDK 源码解析 —— CyclicBarrier

一. 简介 CyclicBarrier 是一个让一系列线程集合互相等待直到一个公共屏障点(barrier point)的同步辅助工具.这个屏障被称为循环屏障,是因为它可以在等待线程释放后被重用. CyclicBarrier 支持一个可选的 Runnable 命令,在最后一个线程到达后执行一次 Runnable 命令. 二. 简单使用示例 CyclicBarrier(3) 等到 3 个线程都到了,这个对象还可以重用,而 CountDownLatch 则不能重用,从 Cyclic 名字就可以看出这个

JDK 源码解析 —— Integer

零. 简介 对于 Integer 这个 Java 程序员几乎天天使用的类, 使用上却可以看出普通程序员和优秀程序员区别. 一. 深入代码 在创建数字 1 的对象时, 大多数人会使用 new Integer(1), 而使用 Integer.valueOf(1) 可以使用系统缓存,既减少可能的内存占用,也省去了频繁创建对象的开销. 系统默认只缓存 -128-127 之间的整数.下面我们看一下 Integer.valueOf(int) 方法的代码: public static Integer valu

JDK 源码解析 —— Semaphore

零. 简介 这是一个用来对并发计数的信号量,并发量超过一定数值则只能等待.从概念上来说,semaphore 维持着一组许可证.获取锁的时候,需要先获得 semaphore 的许可才行. 一. 从 Demo 解析源码 package com.wenniuwuren.concurrent; import java.util.concurrent.ExecutorService; import java.util.concurrent.Executors; import java.util.concu