工厂方法模式【Factory Method Pattern】

这篇具有很好参考价值的文章主要介绍了工厂方法模式【Factory Method Pattern】。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

 前言

1.工厂模式概念


实例化对象,用工厂方法代替new操作(重点)
工厂模式包括工厂方法模式和抽象工厂模式
抽象工厂模式是工厂方法模式的扩展

2.什么情况下适合工厂模式

有一组类似的对象需要创建
在编码时不能预见需要创建哪种类的实例
系统需要考虑扩展性,不应依赖于产品类实例如果被创建、组合和表达的细节

女娲补天的故事大家都听说过吧,今天不说这个,说女娲创造人的故事,可不是“造人”的工作,这
个词被现代人滥用了。这个故事是说,女娲在补了天后,下到凡间一看,哇塞,风景太优美了,天空是湛
蓝的,水是清澈的,空气是清新的,太美丽了,然后就待时间长了就有点寂寞了,没有动物,这些看的到
都是静态的东西呀,怎么办?
别忘了是神仙呀,没有办不到的事情,于是女娲就架起了八卦炉(技术术语:建立工厂)开始创建人,
具体过程是这样的:先是泥巴捏,然后放八卦炉里烤,再扔到地上成长,但是意外总是会产生的:
第一次烤泥人,兹兹兹兹~~,感觉应该熟了,往地上一扔,biu~,一个白人诞生了,没烤熟!
第二次烤泥人,兹兹兹兹兹兹兹兹~~,上次都没烤熟,这次多烤会儿,往地上一扔,嘿,熟过头了,
黑人哪!
第三次烤泥人,兹~兹~兹~,一边烤一边看着,嘿,正正好,Perfect!优品,黄色人类! 
这个过程还是比较有意思的,先看看类图: 
工厂方法模式【Factory Method Pattern】,设计模式,工厂方法模式

 根据类图可以知道需要准备的类:

1.人类接口和对应实现

package factory1;

/**
 * 定义一个人类接口
 * 首先定义什么是人类
 *
 * @author 1
 */
public interface Human {


    /**
     * 人是愉快的,会笑的,本来是想用smile表示,想了一下laugh更合适,好长时间没有大笑了;
     */
    public void laugh();

    /**
     * 人类还会哭,代表痛苦
     */
    public void cry();

    /**
     * 人类会说话
     */
    public void talk();
}
package factory1;

/**
 * 黄种人
 *
 * @author 1
 * @version 1.0
 * @description: TODO
 * @date 2023/8/15 17:15
 */
public class YellowHuman implements Human {
    @Override
    public void laugh() {
        System.out.println("黄色人类会大笑,幸福呀!");
    }

    @Override
    public void cry() {
        System.out.println("黄色人类会哭");
    }

    @Override
    public void talk() {
        System.out.println("黄色人类会说话,一般说的都是双字节");
    }
}

 

package factory1;

/**
 * 白种人
 *
 * @author 1
 * @version 1.0
 * @description: TODO
 * @date 2023/8/15 17:16
 */
public class WhiteHuman implements Human {
    @Override
    public void laugh() {
        System.out.println("白色人类会大笑,侵略的笑声");
    }

    @Override
    public void cry() {
        System.out.println("白色人类会哭");
    }

    @Override
    public void talk() {
        System.out.println("白色人类会说话,一般都是但是单字节!");
    }
}
package factory1;

/**
 * 黑人
 *
 * @author 1
 * @version 1.0
 * @description: TODO
 * @date 2023/8/15 17:17
 */
public class BlackHuman implements Human {
    @Override
    public void laugh() {
        System.out.println("黑人会笑");
    }

    @Override
    public void cry() {
        System.out.println("黑人会哭");
    }

    @Override
    public void talk() {
        System.out.println("黑人可以说话,一般人听不懂");
    }
}

 2.八卦炉

package factory1;

import java.util.List;
import java.util.Random;

/**
 * 造人工厂
 * 女娲造人的故事:
 *
 * @author 1
 * @version 1.0
 * @description: 很久很久以前,盘古开辟了天地,用身躯造出日月星辰、山川草木,天地一片繁华
 * 一天,女娲下界走了一遭,哎!太寂寞,太孤独了,没个会笑的、会哭的、会说话的东东
 * 那怎么办呢?不用愁,女娲,神仙呀,造出来呀,然后捏泥巴,放八卦炉(后来这个成了太白金星的宝
 * 贝)中烤,于是就有了人:
 * 我们把这个生产人的过程用Java程序表现出来:
 * @date 2023/8/15 17:19
 */
public class HumanFactory {

    /**
     * 定义一个八卦炉子
     *
     * @param c
     * @return
     */
    public static Human createHuman(Class c) {
        //定义一个类型的人类
        Human human = null;
        try {
            //生产一个人类
            human = (Human) Class.forName(c.getName()).newInstance();
        } catch (InstantiationException e) {
            System.out.println("必须指定人类颜色");
        } catch (IllegalAccessException e) {
            System.out.println("定义的人类错误");
        } catch (ClassNotFoundException e) {
            System.out.println("定义的人类找不到");
        }
        return human;
    }

    /**
     * 女娲生气了,把一团泥巴塞到八卦炉,哎产生啥人类就啥人类
     *
     * @return
     */
    public static Human createHuman() {
        //定义一个类型的人类
        Human human = null;

        //首先是获得有多少个实现类,多少个人类
        //定义了多少人类
        List<Class> concreteHumanList =
                ClassUtils.getAllClassByInterface(Human.class);
        //八卦炉自己开始想烧出什么人就什么人
        Random random = new Random();
        int rand = random.nextInt(concreteHumanList.size());

        human = createHuman(concreteHumanList.get(rand));

        return human;
    }
}

批量工具类:主要实现根据接口查找实现类(属于扩展知识)

package factory1;

import java.io.File;
import java.io.IOException;
import java.net.URL;
import java.util.ArrayList;
import java.util.Enumeration;
import java.util.List;

/**
 * @author 1
 * @version 1.0
 * @description: TODO
 * @date 2023/8/15 17:48
 */
@SuppressWarnings("all")
public class ClassUtils {
    //给一个接口,返回这个接口的所有实现类
    public static List<Class> getAllClassByInterface(Class c){
        List<Class> returnClassList = new ArrayList<Class>(); //返回结果

        //如果不是一个接口,则不做处理
        if(c.isInterface()){
            String packageName = c.getPackage().getName(); //获得当前的包名
            try {
                //获得当前包下以及子包下的所有类
                List<Class> allClass = getClasses(packageName);

                //判断是否是同一个接口
                for(int i=0;i<allClass.size();i++){
                    if(c.isAssignableFrom(allClass.get(i))){ //判断是不是一个接口
                        if(!c.equals(allClass.get(i))){ //本身不加进去
                            returnClassList.add(allClass.get(i));
                        }
                    }
                }
            } catch (ClassNotFoundException e) {
                e.printStackTrace();
            } catch (IOException e) {
                e.printStackTrace();
            }

        }
        return returnClassList;
    }

    //从一个包中查找出所有的类,在jar包中不能查找
    private static List<Class> getClasses(String packageName)
            throws ClassNotFoundException, IOException {
        ClassLoader classLoader = Thread.currentThread()
                .getContextClassLoader();
        String path = packageName.replace('.', '/');
        Enumeration<URL> resources = classLoader.getResources(path);
        List<File> dirs = new ArrayList<File>();
        while (resources.hasMoreElements()) {
            URL resource = resources.nextElement();
            dirs.add(new File(resource.getFile()));
        }
        ArrayList<Class> classes = new ArrayList<Class>();
        for (File directory : dirs) {
            classes.addAll(findClasses(directory, packageName));
        }
        return classes;
    }


    private static List<Class> findClasses(File directory, String packageName)
            throws ClassNotFoundException {
        List<Class> classes = new ArrayList<Class>();
        if (!directory.exists()) {
            return classes;
        }
        File[] files = directory.listFiles();
        for (File file : files) {
            if (file.isDirectory()) {
                assert !file.getName().contains(".");
                classes.addAll(findClasses(file, packageName + "." +
                        file.getName()));
            } else if (file.getName().endsWith(".class")) {
                classes.add(Class.forName(packageName + '.' +
                        file.getName().substring(0, file.getName().length() - 6)));
            }
        }
        return classes;
    }
}

女娲类

 

package factory1;

/**
 * 女娲
 *
 * @author 1
 * @version 1.0
 * @description: TODO
 * @date 2023/8/15 17:28
 */
public class NvWa {
    public static void main(String[] args) {
        System.out.println("(1)女娲第一次造人,火候不足,白人----------------");
        Human witheHuman = HumanFactory.createHuman(WhiteHuman.class);
        witheHuman.talk();
        witheHuman.laugh();
        witheHuman.cry();


        System.out.println("(2)女娲第二次造人,火候加足点,然后又出了个次品,黑人------------------");
        Human blackHuman = HumanFactory.createHuman(BlackHuman.class);
        blackHuman.cry();
        blackHuman.laugh();
        blackHuman.talk();

        System.out.println("(3)女娲第三次造人,这次火候掌握的正好,黄色人类-----------------");
        Human yellowHuman = HumanFactory.createHuman(YellowHuman.class);
        yellowHuman.cry();
        yellowHuman.laugh();
        yellowHuman.talk();

        System.out.println("(4)女娲让八卦炉随机造人------------------");
        for (int i = 0; i < 10; i++) {
            System.out.println("随机产生人类*******" + i);
            Human human = HumanFactory.createHuman();
            human.cry();
            human.laugh();
            human.talk();
        }
    }
}

运行结果文章来源地址https://www.toymoban.com/news/detail-656603.html

(1)女娲第一次造人,火候不足,白人----------------
白色人类会说话,一般都是但是单字节!
白色人类会大笑,侵略的笑声
白色人类会哭
(2)女娲第二次造人,火候加足点,然后又出了个次品,黑人------------------
黑人会哭
黑人会笑
黑人可以说话,一般人听不懂
(3)女娲第三次造人,这次火候掌握的正好,黄色人类-----------------
黄色人类会哭
黄色人类会大笑,幸福呀!
黄色人类会说话,一般说的都是双字节
(4)女娲让八卦炉随机造人------------------
随机产生人类*******0
白色人类会哭
白色人类会大笑,侵略的笑声
白色人类会说话,一般都是但是单字节!
随机产生人类*******1
黄色人类会哭
黄色人类会大笑,幸福呀!
黄色人类会说话,一般说的都是双字节
随机产生人类*******2
白色人类会哭
白色人类会大笑,侵略的笑声
白色人类会说话,一般都是但是单字节!
随机产生人类*******3
黑人会哭
黑人会笑
黑人可以说话,一般人听不懂
随机产生人类*******4
黑人会哭
黑人会笑
黑人可以说话,一般人听不懂
随机产生人类*******5
黑人会哭
黑人会笑
黑人可以说话,一般人听不懂
随机产生人类*******6
白色人类会哭
白色人类会大笑,侵略的笑声
白色人类会说话,一般都是但是单字节!
随机产生人类*******7
黑人会哭
黑人会笑
黑人可以说话,一般人听不懂
随机产生人类*******8
黑人会哭
黑人会笑
黑人可以说话,一般人听不懂
随机产生人类*******9
白色人类会哭
白色人类会大笑,侵略的笑声
白色人类会说话,一般都是但是单字节!

到了这里,关于工厂方法模式【Factory Method Pattern】的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • 设计模式——抽象工厂模式(Abstract Factory Pattern)

    概述        抽象工厂模式的基本思想是将一些相关的产品组成一个“产品族”,由同一个工厂统一生产。在工厂方法模式中具体工厂负责生产具体的产品,每一个具体工厂对应一种具体产品,工厂方法具有唯一性,一般情况下,一个具体工厂中只有一个或者一组重载的工

    2024年02月03日
    浏览(32)
  • 大话设计模式——2.简单工厂模式(Simple Factory Pattern)

    定义:又称静态工厂方法,可以根据参数的不同返回不同类的实例,专门定义一个类(工厂类)来负责创建其他类的实例可通过类名直接调用,被创建的实例通常具有共同的父类。 UML图: 例子: 计算器中的加减乘除,可将不同的运算看成不同的对象,通过工厂类进行构建,

    2024年02月22日
    浏览(32)
  • 设计模式-抽象工厂模式(Abstract Factory Pattern)结构|原理|优缺点|场景|示例

     目录         设计模式(分类)        设计模式(六大原则)        创建型         工厂方法         抽象工厂模式        单例模式        建造者模式        原型模式      结构型         适配器模式        装饰器模式      

    2024年04月22日
    浏览(32)
  • Simple Factory Pattern 简单工厂模式简介与 C# 示例【创建型】【设计模式来了】

    一句话解释:   客户类和工厂类严格分工,客户类只需知道怎么用,处理逻辑交给工厂类。 简单工厂模式(Simple Factory Pattern)是日常开发中常用的设计模式。其是一种简单的创建型模式,它通过一个工厂类来创建对象,客户端只需要知道如何使用工厂类,而不需要知道对

    2024年02月06日
    浏览(39)
  • Abstract Factory Pattern 抽象工厂模式简介与 C# 示例【创建型】【设计模式来了】

    一句话解释:   提供一个接口,以创建一系列相关或相互依赖的抽象对象,而无需指定它们具体的类。 (将一系列抽象类装进接口,一次接口实现,就必须实例化这一系列抽象类) 抽象工厂模式(Abstract Factory Pattern)是一种创建型模式。它用于创建一组相关对象的家族。

    2024年02月07日
    浏览(32)
  • 【设计模式】模板方法模式(Template Method Pattern)

    模板方法模式是一种行为型设计模式,它定义了一个算法骨架,将某些算法步骤的实现延迟到子类中。 这样可以使得算法的框架不被修改,但是具体的实现可以根据需要进行调整。 在模板方法模式中,我们通常会定义 一个抽象类 ,它包含了 一个模板方法 和 一些抽象方法

    2024年02月12日
    浏览(29)
  • 设计模式-8--模板方法模式(Template Method Pattern)

    模板方法模式(Template Method Pattern)是一种行为型设计模式,它定义了一个算法的骨架,将一些步骤的实现延迟到子类中。模板方法模式允许在不改变算法的结构的情况下,通过在子类中重写特定步骤的具体实现,来改变算法的部分行为。 模板方法模式通常包括以下几个角色

    2024年02月09日
    浏览(26)
  • 设计模式二十三:模板方法模式(Template Method Pattern)

    定义了一个算法的框架,将算法的具体步骤延迟到子类中实现。这样可以在不改变算法结构的情况下,允许子类重写算法的特定步骤以满足自己的需求 模版方法使用场景 算法框架固定,但具体步骤可以变化:当你有一个算法的整体结构是固定的,但其中某些步骤的实现可能

    2024年02月11日
    浏览(26)
  • 工厂方法模式(Factory Method)

    虚拟构造函数(Virtual Constructor)。 工厂方法 是一种创建型设计模式, 其在父类中提供一个创建对象的方法,允许子类决定实例化对象的类型。 1. 问题 假设你正在开发一款物流管理应用。最初版本只能处理卡车运输,因此大部分代码都在位于名为“卡车”的类中。 一段时间

    2024年02月10日
    浏览(36)
  • 设计模式-工厂模式Factory

    一般情况下,工厂模式分为三种更加细分的类型:简单工厂、工厂方法和抽象工厂。 简单工厂叫作静态工厂方法模式(Static Factory Method Pattern) 现在有一个场景,需要一个资源加载器,要 根据不用的url 进行资源加载,但是如果我们将 所有的加载实现代码全部封装在了一个

    2024年02月09日
    浏览(29)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包