彻底玩转抽象工厂模式

一.基础概念

1.定义:

        抽象工厂模式提供了一个创建一系列相关或者相互依赖对象的接口,无需指定它们具体的类。(比如,等会需要定义一个手机接口和一个路由器接口,手机和路由器可以由很多厂商去实现,小米可以实现,华为也是可以实现的,这样我们就无需指定它们的具体类,我们只需要管这个产品就好了,这个定义比较抽象,所以叫抽象工厂)

2.适用场景:

        (1)客户端(应用层)不依赖于产品类实例如何被创建、实现等细节;(比如,手机就是一个产品,我们根本不需要关心它怎么创建的,它只要能打电话就行,很多厂商都去实现了,具体实现的厂商就相当于工厂,手机这个产品就一个抽象的)

        (2)强调一系列相关的产品对象(属于同一产品族)一起使用创建对象需要大量的重复代码;(比如,都是手机,则属于同一产品族,都是路由器,则也属于同一产品族,不同厂商只需都去实现手机、路由器的接口就行了)

        (3)提供一个产品类的库,所有产品以同样的接口出现,从而使得客户端不依赖于具体实现。(比如,所有的手机都需要充电器接口,这就是“所有的产品以同样的接口出现”,客户端去访问的时候就不需要管是谁实现了,它知道这个抽象的接口也是可以做到这个事的)

3.优点:

        (1)具体产品在应用层的代码隔离,无需关心创建的细节

        (2)将一个系列的产品统一到一起创建

        (我们不需要关心是怎么创建的,只需要找到对于工厂拿取即可)

4.缺点:

        (1)规定了所有可能被创建的产品集合,产品簇中扩展新的产品困难;(比如,还需要创建新的产品,就需要修改所有的实现类)

        (2)增加了系统的抽象性和理解难度

5.图形流程:

Creator:相当于抽象工厂,只负责定义两个产品,什么都不需要实现;

ConcreteCreator:相当于工厂,真正的实现交给这些工厂去实现;

6.产品族和产品等级:

(1)小米手机和华为手机都是手机,这些具体的手机和抽象的手机概念就构成了同一产品等级结构,路由器同理也构成了同一产品等级结构;

(2)小米手机是属于手机产品系,它是一个具体的实现,小米路由器也是路由器产品的具体实现;

(3)小米手机和小米路由器虽然属于不同的产品等级结构,但是它们都属于同一个产品族(小米),它都是由小米工厂生产的。

二.具体实现

准备如下几个类:

// 手机产品接口
public interface IphoneProduct {
    void start();
    void shutdown();
    void callup();
    void sendSMS();
}
// 路由器产品接口
public interface IRouterProduct {
    void start();
    void shutdown();
    void openWife();
    void setting();
}
// 小米手机
public class XiaomiPhone implements IphoneProduct{
    @Override
    public void start() {
        System.out.println("开启小米手机");
    }

    @Override
    public void shutdown() {
        System.out.println("关闭小米手机");
    }

    @Override
    public void callup() {
        System.out.println("小米打电话");
    }

    @Override
    public void sendSMS() {
        System.out.println("小米发短信");
    }
}
// 华为手机
public class HuaweiPhone implements IphoneProduct{
    @Override
    public void start() {
        System.out.println("开启华为手机");
    }

    @Override
    public void shutdown() {
        System.out.println("关闭华为手机");
    }

    @Override
    public void callup() {
        System.out.println("华为打电话");
    }

    @Override
    public void sendSMS() {
        System.out.println("华为发短信");
    }
}
// 小米路由器
public class XiaomiRouter implements IRouterProduct{
    @Override
    public void start() {
        System.out.println("启动小米路由器");
    }

    @Override
    public void shutdown() {
        System.out.println("关闭小米路由器");
    }

    @Override
    public void openWife() {
        System.out.println("打开小米wifi");
    }

    @Override
    public void setting() {
        System.out.println("小米设置");
    }
}
// 华为路由器
public class HuaweiRouter implements IRouterProduct{
    @Override
    public void start() {
        System.out.println("启动华为路由器");
    }

    @Override
    public void shutdown() {
        System.out.println("关闭华为路由器");
    }

    @Override
    public void openWife() {
        System.out.println("打开华为wifi");
    }

    @Override
    public void setting() {
        System.out.println("华为设置");
    }
}

XiaomiPhone类和HuaweiPhone类、XiaomiRouter类和HuaweiRouter类都属于同一个产品等级结构;
XiaomiPhone类和XiaomiRouter类、HuaweiPhone类HuaweiRouter类都属于同一个产品族。 

接下来,编写抽象功产品工厂:

// 抽象产品工厂
public interface IProductFactory {
    // 生产手机
    IphoneProduct iphoneProduct();
    // 生产路由器
    IRouterProduct routerProduct();
}

生产手机、路由器只需要返回一个各自的工厂即可,怎么生产的我们不需要管;也可以理解抽象的工厂生产抽象的产品(抽象的抽象),然后这个抽象的产品可以由不同的工厂去实现,就出现了小米工厂和华为工厂,代码如下:

// 小米工厂
public class XiaomiFactory implements IProductFactory{
    @Override
    public IphoneProduct iphoneProduct() {
        return new XiaomiPhone();
    }

    @Override
    public IRouterProduct routerProduct() {
        return new XiaomiRouter();
    }
}
// 华为工厂
public class HuaweiFactory implements IProductFactory{
    @Override
    public IphoneProduct iphoneProduct() {
        return new HuaweiPhone();
    }

    @Override
    public IRouterProduct routerProduct() {
        return new HuaweiRouter();
    }
}

接下来进行测试:

public class Client {
    public static void main(String[] args) {
        System.out.println("================小米系列产品================");
        // 小米工厂
        XiaomiFactory xiaomiFactory = new XiaomiFactory();
        IphoneProduct iphoneProduct = xiaomiFactory.iphoneProduct();
        iphoneProduct.callup(); // 小米打电话
        iphoneProduct.sendSMS(); // 小米发短信
        IRouterProduct iRouterProduct = xiaomiFactory.routerProduct();
        iRouterProduct.openWife(); // 打开小米wifi

        System.out.println("================华为系列产品================");
        // 华为工厂
        HuaweiFactory huaweiFactory = new HuaweiFactory();
        iphoneProduct = huaweiFactory.iphoneProduct();
        iphoneProduct.callup(); // 打开小米wifi
        iphoneProduct.sendSMS(); // 打开小米wifi
        iRouterProduct = huaweiFactory.routerProduct();
        iRouterProduct.openWife(); // 打开小米wifi
    }
}

虽然从小米工厂变成了华为工厂,但是它下面的具体实现都是一样的,我们无需关心。

到处,所以代码都写完了,然后再来理解一下上面的流程图:

如果我们现在还需要生产一个笔记本,只需要在IProductFactory 新增加一个笔记本,但是这样操作违反了开闭原则,但是如果这种修改是长期稳定的,如果增加完不会有变动,也是可以接受的。

 三.工厂模式的选择

1.简单工厂模式、工厂方法模式、抽象工厂模式的具体选择需要根据实际业务来;

2.简单工厂模式和工厂方法模式都关注在产品等级上

3.抽象工厂模式更像是关注一个产品族,对于产品线比较稳定的话,抽象工厂更有效率,毕竟它不是一个工厂生产一个产品,而是一个工厂生产很多产品的产品族,如果不稳定的话,简单工厂更加灵活。

四.使用场景 

1.JDK中Calendar的getInstance方法;

2.JDBC中的Connection对象的获取;

3.Spring中IOC容器创建管理bean对象;

4.反射中Class对象的newInstance方法 。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值