抽象工厂模式

Posted 程姐

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了抽象工厂模式相关的知识,希望对你有一定的参考价值。

抽象工厂模式(Abstract Factory),提供一个创建一系列相关或相互依赖的接口,而无需指定它们具体的类。

涉及到多个产品系列的问题,工厂方法模式组合叫抽象工厂模式。如下图涉及到大于1个的产品,Oracle、mysql两个产品。应用的具体工厂是由两种不同的数据库实现,可以很方便的切换两个数据库的访问的代码。

 

只是涉及到一个产品系列问题,才叫工厂方法模式。如下图 只涉及Leifeng这一产品

抽象工厂模式代码展示开始

Department类

package abstractFactory.dept;

public class Department {
    private long id;
    private String deptName;
    public long getId() {
        return id;
    }
    public void setId(long id) {
        this.id = id;
    }
    public String getDeptName() {
        return deptName;
    }
    public void setDeptName(String deptName) {
        this.deptName = deptName;
    }
    
}

IDeptDao接口类

package abstractFactory.dept;

public interface IDeptDao {
    void insert(Department dept);
    void getDept(int id);
}

MysqlDeptDao类

package abstractFactory.dept;

public class MysqlDeptDao implements IDeptDao {

    @Override
    public void insert(Department dept) {
        System.out.println("在mysql数据库的department表插入一条记录");
    }

    @Override
    public void getDept(int id) {
        System.out.println("在mysql数据库的department表根据ID获取一条记录");
    }

}

OracleDeptDao类

package abstractFactory.dept;

public class OracleDeptDao implements IDeptDao {

    @Override
    public void insert(Department dept) {
        System.out.println("在oracle数据库的department表插入一条记录");
    }

    @Override
    public void getDept(int id) {
        System.out.println("在oracle数据库的department表根据ID获取一条记录");
    }

}

User类

package abstractFactory.user;

public class User {
    private long id;
    private String name;
    public long getId() {
        return id;
    }
    public void setId(long id) {
        this.id = id;
    }
    public String getName() {
        return name;
    }
    public void setName(String name) {
        this.name = name;
    }
    
}

 

IUserDao类

package abstractFactory.user;

public interface IUserDao {
    void insert(User user);
    void getUser(int id);
}

MysqlUserDao类

package abstractFactory.user;

public class MysqlUserDao implements IUserDao {

    @Override
    public void insert(User user) {
        System.out.println("在mysql数据库的user表插入一条记录");
    }

    @Override
    public void getUser(int id) {
        System.out.println("在mysql数据库的user表根据ID获取一条记录");
    }

}

OracleUserDao类

package abstractFactory.user;

public class OracleUserDao implements IUserDao {

    @Override
    public void insert(User user) {
        System.out.println("在oracle数据库的user表插入一条记录");
    }

    @Override
    public void getUser(int id) {
        System.out.println("在oracle数据库的user表根据ID获取一条记录");
    }

}

IFactory类

package abstractFactory;

import abstractFactory.dept.IDeptDao;
import abstractFactory.user.IUserDao;

public interface IFactory {
    IUserDao createUser();
    IDeptDao createDept();
}

MysqlFactory类

package abstractFactory;

import abstractFactory.dept.IDeptDao;
import abstractFactory.dept.MysqlDeptDao;
import abstractFactory.user.IUserDao;
import abstractFactory.user.MysqlUserDao;

public class MysqlFactory implements IFactory {

    @Override
    public IUserDao createUser() {
        return new MysqlUserDao();
    }

    @Override
    public IDeptDao createDept() {
        return new MysqlDeptDao();
    }

}

OracleFactory类

package abstractFactory;

import abstractFactory.dept.IDeptDao;
import abstractFactory.dept.OracleDeptDao;
import abstractFactory.user.IUserDao;
import abstractFactory.user.OracleUserDao;

public class OracleFactory implements IFactory {

    @Override
    public IDeptDao createDept() {
        return new OracleDeptDao();
    }

    @Override
    public IUserDao createUser() {
        return new OracleUserDao();
    }

}

Business类

package abstractFactory;

import abstractFactory.dept.Department;
import abstractFactory.dept.IDeptDao;
import abstractFactory.user.IUserDao;
import abstractFactory.user.User;

public class Business {

    public static void main(String[] args) {
        //产品1系列
        User user = new User();
        IFactory factory = new OracleFactory();
        //IFactory factory = new MysqlFactory();
        IUserDao iuDao = factory.createUser();
        iuDao.insert(user);
        iuDao.getUser(1);
        
        //产品2系列
        Department dept = new Department();
        IDeptDao idepatDao = factory.createDept();
        idepatDao.insert(dept);
        idepatDao.getDept(1);
        
    }

}

 抽象工厂模式的优点和缺点:

最大的好处就是易于交换产品系列,由于具体工厂类,例如:Ifactory factory = new OracleFactory()(或IFactory factory = new MysqlFactory()),在一个应用中只需要在初始化的时候出现一次,这就使得改变一个应用的具体工厂变得非常容易,它只需要改变具体工厂即使用不同的产品配置。

第二大好处:它让具体的创建实例过程与客户端分离,客户端是通过它们的抽象接口操纵实例,产品得具体类名也被具体工厂的实现分离。(oracle还是mysql实现的就不知道,抽象接口IFactory的具体的工厂OracleFactory或MysqlFactory实现分离)

缺点:需求是增加功能,增加一个职责(job类),需要增加Job、IJobDao、MysqlJobDao、OracleJobDao这4个类(增加不可避免),同时还要修改IFactory、OracleFactory、MysqlFactory,才能增加职责。修改的类多,而且违反“开放-封闭原则”。

 

以上是关于抽象工厂模式的主要内容,如果未能解决你的问题,请参考以下文章

设计模式学习——简单工厂模式工厂模式抽象工厂模式

设计模式 创建者模式 工厂设计模式 -- 抽象工厂设计模式介绍和实现

设计模式:抽象工厂模式

设计模式11:抽象工厂模式

C++工厂模式(简单工厂工厂方法抽象工厂)

C++工厂模式(简单工厂工厂方法抽象工厂)