对初识Spring工厂模式的思考

Spring的基本理论知识

  • 什么是Spring?

Spring 是分层的 Java SE/EE应用 full-stack 轻量级开源框架,以 IoC(Inverse Of Control:
反转控制)和 AOP(Aspect Oriented Programming:面向切面编程)为内核,提供了表现层 Spring
MVC 和持久层 Spring JDBC 以及业务层事务管理等众多的企业级应用技术,还能整合开源世界众多
著名的第三方框架和类库,逐渐成为使用最多的Java EE 企业应用开源框架。

  • spring 的体系结构
  • Spring的优势?

方便解耦,简化开发
AOP 编程的支持
声明式事务的支持(配置的方式)
方便程序的测试(好像集成了Junit)
方便集成各种优秀框架
降低 JavaEE API
Java 源码是经典学习范例

栗子

我们有这么一个场景,需要实现一个账户保存的功能。这个很简单,无非就是两个接口、两个实现类、再加一个测试类的事情。

创建账户持久层接口 IAccountDao,定义保存账户抽象方法

public interface IAccountDao {
    /**
     * 保存账户
     */
    void saveAccount();
}

实现账户持久层接口 AccountDaoImpl,并创建并实现 saveAccount 方法,输出保存信息。

public class AccountDaoImpl implements IAccountDao {
 @Override
    public void saveAccount() {
        System.out.println("已保存账户~!");
    }
}

定义账户业务层接口 IAccountService,并定义一个保存账户的方法

public interface IAccountService {
    /**
     * 保存账户
     */
    void saveAccount();
}

实现账户业务层接口 AccountServiceImpl,实现调用持久层 saveAccount 方法实现保存账号抽象方法。

public class AccountServiceImpl implements IAccountService {
  /**
     * 持久层对象
     */
    private IAccountDao accountDao = new AccountDaoImpl();

    @Override
    public void saveAccount() {
        accountDao.saveAccount();
    }
}

最后编写表现层 Test,测试下该用例是否能正常运行

public class Test{
    public static void main(String[] args) {
        IAccountService as = new AccountServiceImpl();
        as.saveAccount();
    }
}

// 以下为输出结果
已保存账户~!

分析:

1、首先要清楚一个概念:耦合

  • 程序的耦合

耦合:程序间的依赖关系。包括 类之间的依赖、方法间的依赖。
解耦:降低程序间的依赖关系
实际开发中应该做到编译期不依赖,运行时才依赖。
解耦的思路:
第一步:使用反射来创建对象,而避免使用new关键字。
第二步:通过读取配置文件来获取要创建的对象全限定类名
在实际开发中,耦合是必然存在的,故只能尽量降低依赖,不能消除。

在上面的这几段代码中,耦合程度是很高的。为啥说高耦合呢,假若我们不小心删了 AccountDaoImpl ,整个程序都无法通过编译。那么这时候就需要改写代码,使用工厂模式解耦!!!

2、配置文件的使用

  • 为什么要使用配置文件呢??
    使用配置文件可以达到一个目的,就是我们可以将一些固定不变的内容抽取出来单独保存,通过调用的方式获取其实例。

首先需要一个配置文件来配置我们的service和dao。配置的内容:唯一标识=全限定类名(key=value)
其次通过读取配置文件中配置的内容,反射创建对象。

代码的改写

1、创建 bean.properties 文件,保存账户业务层实现类的路径 AccountServiceImpl 和账户持久层接口实现类 AccountDaoImpl 信息

accountService=service.AccountServiceImpl
accountDao=dao.AccountDaoImpl

2、创建Bean对象的工厂模式 BeanFactory 。Bean:在计算机英语中,有可重用组件的含义;
JavaBean:用java语言编写的可重用组件。它就是创建我们的service和dao对象的。

public class BeanFactory {

    /**
     * Properties 对象
     */
    private static Properties props;

    /**
     * 静态代码块为 Properties 对象赋值
     */
    static {
        try {
            // 实例化对象
            props = new Properties();
            // 使用类加载器获取配置文件位置,获取 properties 文件的流对象
            InputStream in = BeanFactory.class.getClassLoader().getResourceAsStream("bean.properties");
            props.load(in);
        } catch (IOException e) {
            throw new ExceptionInInitializerError("初始化 properties 文件异常!");
        }
    }

    /**
     * 获取对象实例
     * @param beanName 对象实例名
     * @return {@link Object} 对象实例
     */
    public static Object getBean(String beanName) {
        Object bean = null;
        try {
            String beanPath = props.getProperty(beanName);
            //以反射的方式创建对象
            bean = Class.forName(beanPath).newInstance();
        }catch (Exception e) {
            e.printStackTrace();
        }
        return bean;
    }
}

此时,账户业务层实现类就可以这么写:

public class AccountServiceImpl implements IAccountService {
    // 通过工厂模式获取 AccountDaoImpl 的实例
    private IAccountDao accountDao = (IAccountDao) BeanFactory.getBean("accountDao");
    @Override
    public void saveAccount() {
        accountDao.saveAccount();
    }
}

表现层这么写:

public class Test {
    public static void main(String[] args) {
        IAccountService as = (AccountServiceImpl) BeanFactory.getBean("accountService");
        as.saveAccount();
    }
}

在这里插入图片描述
这时候相比之前耦合程度就大大降低了,这时候尽管删除持久层包下的实现类 AccountDaoImpl,编译是不会出错,但是没有该类是肯定运行不了。编译器会报 java.lang.ClassNotFoundException 异常,而不是 Error。
耦合只是相对而言,工厂模式也不是完美的,仍然存在很多可以优化的地方,但是可以减少部分耦合就是对程序性能的一个大幅度的提升。

代码的进一步改写

BeanFactory中有这么一句:bean = Class.forName(beanPath).newInstance();,这就使得每次都会调用默认构造函数创建对象,前提是我们的对象只能newInstance一次,由于创建对象后,不存起来,那么java的垃圾回收机制就会在长时间不用后将其回收,下次再用的时候肯定就没有了,所以这并不是一个单例的。
ps:多例相对于单例来说,对象被创建多次,执行效率没有单例对象高!newinstance()和new的区别?

改写BeanFactory:
思路:当newInstance创建对象后,需要将对象存起来。用什么存?Map!!!

public class BeanFactory {
    //定义一个Properties对象
    private static Properties props;

    //定义一个Map,用于存放我们要创建的对象。我们把它称之为容器
    private static Map<String,Object> beans;

    //使用静态代码块为Properties对象赋值
    static {
        try {
            //实例化对象
            props = new Properties();
            //获取properties文件的流对象
            InputStream in = BeanFactory.class.getClassLoader().getResourceAsStream("bean.properties");
            props.load(in);
            //实例化容器
            beans = new HashMap<String,Object>();
            //取出配置文件中所有的Key
            Enumeration keys = props.keys();
            //遍历枚举
            while (keys.hasMoreElements()){
                //取出每个Key
                String key = keys.nextElement().toString();
                //根据key获取value
                String beanPath = props.getProperty(key);
                //反射创建对象
                Object value = Class.forName(beanPath).newInstance();
                //把key和value存入容器中
                beans.put(key,value);
            }
        }catch(Exception e){
            throw new ExceptionInInitializerError("初始化properties失败!");
        }
    }

    /**
     * 根据bean的名称获取对象
     * @param beanName
     * @return
     */
    public static Object getBean(String beanName){
        return beans.get(beanName);
    }

}

  • 1
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值