手写springiocxml方式注入对象

什么是SpringIOC
spring ioc指的是控制反转,IOC容器负责实例化、定位、配置应用程序中的对象及建立这些对象间的依赖。
交由Spring来管理这些,实现解耦
SpringIOC原理
使用反射机制+XML技术

 

<beans xmlns="http://www.springframework.org/schema/beans"
	xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:p="http://www.springframework.org/schema/p"
	xmlns:context="http://www.springframework.org/schema/context"
	xmlns:aop="http://www.springframework.org/schema/aop" xmlns:tx="http://www.springframework.org/schema/tx"
	xsi:schemaLocation="http://www.springframework.org/schema/beans
    	 http://www.springframework.org/schema/beans/spring-beans.xsd
     	 http://www.springframework.org/schema/context
         http://www.springframework.org/schema/context/spring-context.xsd
         http://www.springframework.org/schema/aop
         http://www.springframework.org/schema/aop/spring-aop.xsd
         http://www.springframework.org/schema/tx
     	 http://www.springframework.org/schema/tx/spring-tx.xsd">
     	 
	
	<!-- 把对象初始化之后,存在HashMap里面去 
		放到内存里面去,JVM内存里面去之后,别人只要调用getBean方法之后,去Spring的容器里面去找到一个对象就行了
		这种方式也可以,这种方式我待会再讲,再讲注解的时候我会去实现,所以我是把两种方式都会实现的,待会我讲注解方式的时候会
		详细讲的,在Spring中id如果重复的情况下是会报错的,所以有的时候是需要做校验的,找最后默认的也可以,这个看你自己
		怎么实现,
	-->
	<bean id="user" class="com.learn.spring.entity.User"></bean>
	
</beans>
package com.learn.spring.entity;

public class User {

	private Integer id;
	private String userName;

	public Integer getId() {
		return id;
	}

	public void setId(Integer id) {
		this.id = id;
	}

	public String getUserName() {
		return userName;
	}

	public void setUserName(String userName) {
		this.userName = userName;
	}

}
package com.learn.spring;

import java.io.InputStream;
import java.util.List;

import org.apache.commons.lang.StringUtils;
import org.dom4j.Document;
import org.dom4j.DocumentException;
import org.dom4j.Element;
import org.dom4j.io.SAXReader;

/**
 * 我们先定义这个类
 * 我会将两种方式的实现方式
 * 我们建完ClassPathXmlApplicationContext这个类
 * 自定义Spring容器框架
 * xml方式实现
 * 你们在写代码之前
 * 不要盲目的去写
 * 先通过白话文
 * 他不知道思路
 * 你先通过白话文知道整个实现过程
 * 那这个我们怎么实现呢
 * 
 * 
 * @author Leon.Sun
 *
 */
public class ClassPathXmlApplicationContext {
	// xml路径地址
	/**
	 * xml读取路径地址
	 * 
	 */
	private String xmlPath;

	/**
	 * 定义一个构造函数
	 * 传入xmlPath
	 * 
	 * 
	 * @param xmlPath
	 */
	public ClassPathXmlApplicationContext(String xmlPath) {
		this.xmlPath = xmlPath;
	}

	/**
	 * 这里要传入一个bean的id过来
	 * 返回的是一个Object对象
	 * 这个方法也返回Object对象
	 * 
	 * 
	 * @param beanId
	 * @return
	 * @throws Exception
	 */
	public Object getBean(String beanId) throws Exception {
		// 1. 读取配置文件
		/**
		 * 解析XML文件
		 * 如果不解析
		 * 我怎么去读取这个bean
		 * 是不是这样的
		 * 这肯定要解析
		 * 解析是第一步
		 * 第一步是把XML文件解析一遍
		 * 
		 */
		List<Element> elements = readerXml();
		/**
		 * 还有一种场景就是我没有配置任何bean
		 * 代码根本就不用往下面再走了
		 * 是不是这样的
		 * 我们是不是可以直接判断一下
		 * 
		 */
		if (elements == null) {
			throw new Exception("该配置文件没有子元素");
		}
		// 2. 使用beanId查找对应的class地址
		/**
		 * 我把xml全部解析了之后
		 * 第二步干嘛
		 * 使用方法参数bean id查找配置文件中bean节点的id信息是否一致
		 * 假设我这个时候传入userService
		 * 这样的一个beanid
		 * 查找到了就要获取他的class地址
		 * 是不是这样的
		 * 然后方法里面传入一个参数过来userService
		 * 我能不能找到userService的一个节点
		 * 是不是找到这个节点
		 * 
		 * 
		 */
		String beanClass = findXmlByIDClass(elements, beanId);
		if (StringUtils.isEmpty(beanClass)) {
			throw new Exception("未找到对应的class地址");
		}
		// 3. 使用反射机制初始化,对象
		/**
		 * 获取class信息地址
		 * 使用反射机制初始化
		 * 
		 * 找到class信息
		 * 拿到信息之后我使用JAVA的反射机制
		 * 初始化之后返回给调用者
		 * 这里有多少个实现方式
		 * 
		 * 
		 */
		Class<?> forName = Class.forName(beanClass);
		return forName.newInstance();
	}

	// 读取配置文件信息
	/**
	 * 表示解析这样的一个XML
	 * 这个传入的可以没有
	 * 因为它直接拿全局的
	 * 这个大家一定要慢慢的去想
	 * 这个讲的非常细
	 * 
	 * 解析XML文件信息
	 * 拿到每个节点的信息
	 * 拿到每个文件信息以后
	 * 传入一个bean的ID去里面去找
	 * 进行反射去实现出来
	 * 你们可以先把这些节点信息给他封装起来
	 * 怎么封装呢
	 * 
	 * 
	 * @return
	 * @throws DocumentException
	 */
	public List<Element> readerXml() throws DocumentException {
		SAXReader saxReader = new SAXReader();
		if (StringUtils.isEmpty(xmlPath)) {
			new Exception("xml路径为空...");
		}
		Document read = saxReader.read(getClassXmlInputStream(xmlPath));
		// 获取根节点信息
		/**
		 * 读取他的根节点
		 * 
		 */
		Element rootElement = read.getRootElement();
		// 获取子节点
		/**
		 * 获取根节点下的所有的子节点
		 * 获取这样的一个子节点之后
		 * 获取项目下的所有的子节点
		 * 获取所有子节点element的对象
		 * 获取完了之后我们再怎么进行实现呢
		 * 这个时候我可以拿到所有的bean的节点
		 * 
		 */
		List<Element> elements = rootElement.elements();
		/**
		 * 就是判断是否为空
		 * 
		 */
		if (elements == null || elements.isEmpty()) {
			return null;
		}
		return elements;
	}

	// 使用beanid查找该Class地址
	/**
	 * 是用beanid去查找有没有这样的class地址
	 * 我们是不是要调用根节点
	 * 
	 * 
	 * 
	 * @param elements
	 * @param beanId
	 * @return
	 * @throws Exception
	 */
	public String findXmlByIDClass(List<Element> elements, String beanId) throws Exception {
		for (Element element : elements) {
			// 读取节点上是否有value
			/**
			 * 获取属性的信息
			 * 这个id就表示当前的userService
			 * 
			 * 
			 */
			String beanIdValue = element.attributeValue("id");
			/**
			 * 如果xml的id等于空的情况下
			 * 就结束当前循环
			 * 你如果使用Spring的时候
			 * 会校验的
			 * 这个具体看你们自己
			 * 
			 */
			if (beanIdValue == null) {
				throw new Exception("使用该beanId为查找到元素");
			}
			if (!beanIdValue.equals(beanId)) {
				/**
				 * 结束本次循环
				 * 
				 */
				continue;
			}
			// 获取Class地址属性
			/**
			 * 拿到这样的一个class地址
			 * 拿到之后
			 * 然后用反射去做一个初始化过程
			 * 
			 */
			String classPath = element.attributeValue("class");
			/**
			 * 如果等于null的情况下它会怎么样
			 * 
			 */
			if (!StringUtils.isEmpty(classPath)) {
				return classPath;
			}
		}
		return null;
	}

	// 读取xml配置文件
	/**
	 * 获取当前上下文路径
	 * 
	 * @param xmlPath
	 * @return
	 */
	public InputStream getClassXmlInputStream(String xmlPath) {
		InputStream resourceAsStream = getClass().getClassLoader().getResourceAsStream(xmlPath);
		return resourceAsStream;
	}

}
package com.learn.spring;

import com.learn.spring.entity.User;

public class Test002 {

	public static void main(String[] args) throws Exception {
		ClassPathXmlApplicationContext applicationContext = new ClassPathXmlApplicationContext("spring.xml");
		User user = (User) applicationContext.getBean("user");
		System.out.println(user);
	}

}

 

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

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值