引入

  • 假设有一个司机, 需要到某个城市, 于是我们给他一辆汽车
public class Demo {
public static void main(String[] args) {
Car car = new Car();
car.run();
}
} public class Car {
public void run(){
System.out.println("汽车正在向前跑...");
}
}
  • 如果我们希望给到这个司机的始终是一辆车, 应该怎么做? (单例)
  • 首先我们不能让司机自己通过new产生一辆汽车, 而是应该通过调用Car类中的某个方法对外提供车.
public class Car {
private static Car car = new Car();//用于提供给外界, 始终是同一辆车 private Car(){};//私有构造方法, 在类之外不能通过new获得本类对象了, 保证了单例 public Car getInstance(){
return car;
} public void run(){
System.out.println("汽车正在向前跑...");
}
} public static void main(String[] args) {
Car car = Car.getInstance();
car.run();
}

 

简单工厂

  • 下面考虑, 如果我们不希望只有汽车这种交通工具, 我们希望可以定制交通工具, 并定制生产交通工具的流程, 应该怎么做?
  • 一旦产生由汽车到交通工具这样的概念, 就应该想到多态. 我们可以定义一个Moveable接口, 在接口中声明run()方法, 所有的交通工具类都实现该接口.
  • 对于定制生产流程, 我们可以通过一个工厂进行生产对应的交通工具.
public interface Moveable {
void run();
} public class Car implements Moveable{ public Car(){};//私有构造方法, 在类之外不能通过new获得本类对象了, 保证了单例 public void run(){
System.out.println("汽车正在向前跑...");
}
} public abstract class VehicleFactory {
public abstract Moveable create();
} public class CarFactory extends VehicleFactory {
@Override
public Moveable create() {
return new Car();
}
} //Test
public static void main(String[] args) {
VehicleFactory factory = new CarFactory();
Moveable m = factory.create();
m.run();
}

 

抽象工厂

  • 下面把简单工厂的画面从脑海中清空, 讲述另一种工厂实现.
  • 我们假设开头的司机不是一个普通的司机, 他除了需要一种交通工具以到达某个城市外, 他还需要一把AK47, 并且还需要一个苹果以备路上不时之需.
  • 所以我们需要给他一个工厂来制造这一系列产品.
  • 为了提高可扩展性, 我们还希望不同的工厂可以制作不同系列的产品, 比如上面说的A工厂制造的是汽车, AK47, 苹果; 而B工厂制造的是飞机, 火箭炮, 旺仔小馒头.
//test
public static void main(String[] args) {
AbstractFactory factory = new Factory1();
Vehiche v = factory.createVehiche();
Weapon w = factory.createWeapon();
Food f = factory.createFood(); v.run();
w.fire();
f.eat();
} public abstract class Vehiche {//交通工具的抽象类
public abstract void run();
} public abstract class Weapon {//武器的抽象类
public abstract void fire();
} public abstract class Food {//食物的抽象类
public abstract void eat();
} public class Car extends Vehiche{一种具体的交通工具
@Override
public void run() {
System.out.println("小汽车启动...");
}
} public class AK47 extends Weapon {//一种具体的武器
@Override
public void fire() {
System.out.println("哒哒哒...");
}
} public class Apple extends Food{//一种具体的食物
@Override
public void eat() {
System.out.println("大口吃苹果...");
}
} //抽象工厂
public abstract class AbstractFactory {
public abstract Vehiche createVehiche();
public abstract Weapon createWeapon();
public abstract Food createFood();
} //抽象工厂的实现1
public class Factory1 extends AbstractFactory {
@Override
public Vehiche createVehiche() {
return new Car();
} @Override
public Weapon createWeapon() {
return new AK47();
} @Override
public Food createFood() {
return new Apple();
}
}

 

  • 总结一下, 抽象工厂和简单工厂各有什么优劣?
  • 抽象工厂能够生产一系列产品, 也能方便地替换掉一系列产品, 但是如果想要在产品系列中添加多一个品种将会非常麻烦. 比如说在上面的系列产品中添加一个盔甲抽象类, 那么抽象工厂以及对应的实现都要修改源码了.
  • 而简单工厂能够灵活的生产但一个品种的产品, 但是如果生产的品种较多, 会出现工厂泛滥的问题.
  • 两者优劣互补, 那么有没有可以兼容两者优点的工厂实现呢? 下面看spring的工厂实现, 它给出了一种解决方案.

 

Spring的bean工厂

  • 我们再次考虑最原始的情况, 有一个Moveable接口, 里面有run方法, Car小汽车类实现了该接口.
public static void main(String[] args) {
Moveable m = new Car();
m.run();
} public interface Moveable {
void run();
} public class Car implements Moveable{
@Override
public void run() {
System.out.println("小汽车往前跑...");
}
}
  • 在Spring的bean工厂中, 新对象不是通过new关键字获取的, 而是通过配置文件获取的.
  • 具体的过程是: 先读取配置文件获得该类的class对象, 然后通过class对象创建具体的实例对象.
public static void main(String[] args) throws Exception {
//获取配置文件
Properties props = new Properties();
props.load(Test.class.getClassLoader().getResourceAsStream("spring.properties"));
//获取配置文件中配置的类
String vehicheTypeName = props.getProperty("vehicheTypeName");
//反射生成对应的对象
Moveable m = (Moveable) Class.forName(vehicheTypeName).newInstance();
m.run();
} //spring.properties
vehicheTypeName=designPattern.factory.springFactory.Car

 

  • 上面是对spring中bean工厂使用的模拟, 下面我们使用真实的spring来生成Car对象, 对比一下.
public static void main(String[] args) throws Exception {
BeanFactory bf = new ClassPathXmlApplicationContext("applicationContext.xml");
Vehiche v = (Vehiche)bf.getBean("v");
v.run();
} //配置文件
<bean id="v" class="designPattern.factory.Car">
</bean>
  • 经过对比我们发现我们自己写的简单工厂和spring的bean工厂在使用上没有什么区别, 确实spring使用起来就是这么简单, 下面我们模拟一下spring的bean工厂实现.

 

模拟Spring工厂实现

模拟IOC

  • 都说spring是个bean容器, 以下的代码将展示它是如何生成bean, 并把bean放入容器中供用户获取的.
  • 思路比较简单:
  1. 创建BeanFactory工厂接口, 添加方法getBean().
  2. 创建BeanFactory的实现类ClassPathXmlApplicationContext. 将在该实现类中展示IOC的具体实现.
  3. ClassPathXmlApplicationContext需要一个container容器存放创建的bean对象, 这里使用HashMap实现.
  4. ClassPathXmlApplicationContext的构造方法中读取spring的配置文件, 这里使用到了dom4j. 读取配置文件后根据beanclass属性使用反射创建出bean对象. 然后把idbean对象分别作为keyvalue添加到容器中.
  5. 当工厂被调用getBean()方法时, 从容器中找到对应的bean并返回.
public static void main(String[] args) throws Exception {
BeanFactory bf = new ClassPathXmlApplicationContext("applicationContext.xml");
Vehiche v = (Vehiche) bf.getBean("v");
v.run();
} //BeanFactory的实现类
public class ClassPathXmlApplicationContext implements BeanFactory { private Map<String, Object> container = new HashMap<>();//用于存放bean对象的容器 //在构造方法中读取xml配置文件, 把bean对象都创建好并放入容器中
public ClassPathXmlApplicationContext(String propAddr) throws Exception {
SAXReader reader = new SAXReader();
File file = new File(this.getClass().getClassLoader().getResource(propAddr).toURI());
Document document = reader.read(file);
Element root = document.getRootElement();
List<Element> childElements = root.elements(); for (Element child : childElements) {
Object bean = Class.forName(child.attributeValue("class")).newInstance();
container.put(child.attributeValue("id"), bean);
}
} @Override
public Object getBean(String beanId) {
return container.containsKey(beanId) ? container.get(beanId) : null;
}
} //极简BeanFactory
public interface BeanFactory {
Object getBean(String beanId);
} //xml中配置的bean
<bean id="v" class="designPattern.factory.Car">
</bean>

工厂模式讲解, 引入Spring IOC的更多相关文章

  1. Java工厂模式解耦 —— 理解Spring IOC

    Java工厂模式解耦 -- 理解Spring IOC 最近看到一个很好的思想来理解Spring IOC,故记录下来. 资源获取方式 主动式:(要什么资源都自己创建) 被动式:(资源的获取不是我们创建, ...

  2. 工厂模式如何返回Spring的Bean

    工厂返回的可以是一个具体的对象,比如造一辆车,可以返回一个自行车对象,或者汽车对象. 但是在Spring 中需要工厂返回一个具体的Service,这就是一个抽象工厂了 一种方法是反射,个人觉得这种方式 ...

  3. Spring IOC 方式结合TESTGN测试用例,测试简单java的命令模式

    java命令模式: 可以命令("请求")封装成一个对象,一个命令对象通过在特定的接收着上绑定一组动作来封装一个请求.命令对象直接把执行动作和接收者包进对象中,只对外暴露出执行方法的 ...

  4. 深入理解Spring IOC容器

    本文将从纯xml模式.xml和注解结合.纯注解的方式讲解Spring IOC容器的配置和相关应用. 纯XML模式 实例化Bean的三种方式: 使用无参构造函数 默认情况下,会使用反射调用无参构造函数来 ...

  5. 深入理解Spring IOC容器及扩展

    本文将从纯xml模式.xml和注解结合.纯注解的方式讲解Spring IOC容器的配置和相关应用. 纯XML模式 实例化Bean的三种方式: 使用无参构造函数 默认情况下,会使用反射调用无参构造函数来 ...

  6. Spring:源码解读Spring IOC原理

    Spring IOC设计原理解析:本文乃学习整理参考而来 一. 什么是Ioc/DI? 二. Spring IOC体系结构 (1) BeanFactory (2) BeanDefinition 三. I ...

  7. Spring IOC原理解读 面试必读

    Spring源码解析:Bean实例的创建与初始化 一. 什么是Ioc/DI? 二. Spring IOC体系结构 (1) BeanFactory (2) BeanDefinition 三. IoC容器 ...

  8. Spring源码解读Spring IOC原理

    一.什么是Ioc/DI? IoC 容器:最主要是完成了完成对象的创建和依赖的管理注入等等. 先从我们自己设计这样一个视角来考虑: 所谓控制反转,就是把原先我们代码里面需要实现的对象创建.依赖的代码,反 ...

  9. Spring IOC设计原理解析:本文乃学习整理参考而来

    Spring IOC设计原理解析:本文乃学习整理参考而来 一. 什么是Ioc/DI? 二. Spring IOC体系结构 (1) BeanFactory (2) BeanDefinition 三. I ...

随机推荐

  1. C语言出来多久了你知道吗?

    在20世纪80年代,为了避免不同开发者使用的C语言语法的差异,美国国家标准局为C语言开发了一套完整的美国国家标准语言文法,称为ANSI C,作为C语言的初始标准.. [1] 2011年12月8日,国际 ...

  2. LinkedList源码

    1.介绍及注意事项 链表由Josh Bloch书写,属于Java集合框架中的一种,LinkedList实现的是双链表,实现了所有的链表操作,可能够实现所有元素(包括)的基本操作. 链表是非线程同步的, ...

  3. mysql 基本命令操作

    1. 查看存储引擎 show engines; 2. 查看数据存储位置 show variables like 'datadir': 3. 存储引擎 create table mytest engin ...

  4. 对Python这门课程的理解。

    这门课程是现在热门,对之后的就业和利用的帮助还是很大的. 希望能学完整本书并且能学以致用,而不是单单只获得理论知识. 学完之后能用于数据库.大数据处理.图形编程等等

  5. vue config.js配置生产环境和发布环境不同的接口地址问题

    第一步,分别设置不同的接口地址 首先,我们分别找到下面的文件: /config/dev.env.js /config/prod.env.js 其实,这两个文件就是针对生产环境和发布环境设置不同参数的文 ...

  6. Python_生成大量随机信息

    #coding=utf-8 import random import string import codecs ''' 演示如何使用Python标准库random来生成随机数据,这在需要 ''' #常 ...

  7. vue项目中解决type=”file“ change事件只执行一次的问题

    问题描述 在最近的项目开发中遇到了这样的一个问题,当我上传了一个文件时,我将获取到的文件名清空后,却无法再次上传相同的文件 <template> <div class="h ...

  8. C语言 > 字符串和字符串函数

    输入 gets() 函数 : 1.gets() 从标准输入设备读取字符串,以回车结束读取,使用'\0'结尾,回车符'\n'被舍弃没有遗留在缓冲区. 2.可以用来输入带空格的字符串. 3.可以无限读取, ...

  9. Hibernate中的持久化类

    一.持久化类概述 就是一个JavaBean,这个JavaBean与表建立了映射关系.这个类就称为是持久化类. 简单理解为 持久化类=JavaBean+映射文件. 持久化类:是指其实例需要被Hibern ...

  10. SSM-SpringMVC-12:SpringMVC中BeanNameViewResolver这种视图解析器

    ------------吾亦无他,唯手熟尔,谦卑若愚,好学若饥------------- 视图解析器,这个很熟悉啊,之间就用过,就是可以简写/和.jsp的InternalResourceViewRes ...