自己动手写spring(一) 使用digester

前言

本来想熟悉下digester的使用,写着写着发现,可以搞一个类似spring的东西的来,将过程记录下来,与大家分享。

例子中很多代码没有优化,基本没有异常处理和判空操作,这是我的一个坏习惯,先奔着目的去,实现成功后再重构。但这样的代码,更易看懂,优化后的代码反而掩盖了很多思路和思想,尤其容易让人分不清重点(因为算是技巧的地方太多了)。

项目源码的git clone地址:git@code.csdn.net:lqk654321/lspring.git

使用digester读取一个配置文件

digester入门的例子通常是

<?xml version="1.0" encoding="UTF-8"?>
<books>
	<book>
		<author>Author 2</author>
		<title>His One Book</title>
	</book>
</books>

digester代码

public class BookMain {
	public static void main(String[] args) throws IOException, SAXException {
		  Digester digester = new Digester();
          digester.setValidating(false);
          // 创建对象
          digester.addObjectCreate("books", Books.class);
          digester.addObjectCreate("books/book", Book.class);
          // 设置属性
          digester.addBeanPropertySetter("books/book/author", "author");
          digester.addBeanPropertySetter("books/book/title", "title");
          // 处理对象之间的关系
          digester.addSetNext("books/book", "addBook");
          InputStream in = ClassLoader.getSystemResourceAsStream("books.xml");
          Books bs = (Books) digester.parse(in);
          System.out.println(bs.getBooks().get(0));
	}
}

这个xml文件的配置跟我们常用的不太一样,很明显,常见的是下面这种:

<books>
	<book author="author" title="title" class="org.lqk.digester.book.Book"/>
</books>

对应的处理类

public class BookMain2 {
	public static void main(String[] args) throws IOException, SAXException {
		  Digester digester = new Digester();
          digester.setValidating(false);
          digester.addObjectCreate("books", Books.class);
          digester.addObjectCreate("books/book", "org.lqk.digester.book.Book","class");
          digester.addSetProperties("books/book");
          digester.addSetNext("books/book", "addBook");
          InputStream in = ClassLoader.getSystemResourceAsStream("books2.xml");
          Books bs = (Books) digester.parse(in);
          System.out.println(bs.getBooks().get(0));
	}
}

这个跟常用的就像很多了。从这个例子中可以看到,根据配置文件,我们是可以搞出一些类来,并处理它们之间的关系。说的抽象点,我们可以将配置文件描述的信息,映射到jvm的类中。

spring样式的配置文件

那对于一个spring样式的配置文件,兴许也能搞出点东西来。

<beans>
	<!-- 目前的局限 
		1. 必须包括id和class两个属性
	-->
	<!-- <bean>标签中的name属性对Bean类有意义,但对beanA和beanB确没有意义 -->
	<bean id="beanA" name="test" class="org.lqk.container.bean.BeanA">
		<property name="beanB" ref="beanB"/>
		<property name="title" value="studentA"/>
	</bean>
	<bean id="beanB" class="org.lqk.container.bean.BeanB">
		<property name="title" value="studentB"/>
	</bean>
</beans>

我们先来尝试读取基本信息。首先是创建三个model,将标签信息对象化(你该不会想直接用digester生成beanA吧!),model中省略set和get方法(但是必须要有的喔)。

public class Beans {
	private List<Bean> beans = new ArrayList<Bean>();
	public void addBean(Bean bean){
		this.beans.add(bean);
	}
}

public class Bean {
	private String className;
	private String id;
	private List<Property> props = new ArrayList<Property>();
	public void addProperty(Property property){
		this.props.add(property);
	}
}	

public class Property {
	private String name;
	private String value;
	private String ref;
}

public class Main {
	public static void main(String[] args) throws IOException, SAXException {
		Digester digester = new Digester();
		digester.setValidating(false);
		digester.addObjectCreate("beans", Beans.class);

		// 如果配置文件中有多个bean,add一次即可
		digester.addObjectCreate("beans/bean", Bean.class);

		// 设置bean的属性<bean name="",id="">中的id和name。默认属性名和类中的属性名一样,不同的要特殊配置
		digester.addSetProperties("beans/bean", "class", "className");
		digester.addSetProperties("beans/bean");

		digester.addObjectCreate("beans/bean/property", Property.class);
		// 将标签中的属性赋给对象
		digester.addSetProperties("beans/bean/property");

		// 设置对象间的关系
		digester.addSetNext("beans/bean/property", "addProperty");
		digester.addSetNext("beans/bean", "addBean");

		InputStream in = ClassLoader.getSystemResourceAsStream("beans.xml");
		Beans beans = (Beans) digester.parse(in);
		List<Bean> beanList = beans.getBeans();
		for (Bean bean : beanList) {
			System.out.println("bean =================================================>");
			System.out.println("    id ==> " + bean.getId());
			List<Property> props = bean.getProps();
			for (Property prop : props) {
				System.out.println("    property =================================================>");
				System.out.println("        name ==> " + prop.getName());
				System.out.println("        ref ==> " + prop.getRef());
				System.out.println("        value ==> " + prop.getValue());
			}
		}
	}
}

现在信息从配置文件转到了jvm中,以类的形式存在。我们可以通过Class.forName(className)加载真正的beanA进入jvm。柿子捡软的捏,我们先创建一个beanB出来。

public class Main2 {	
	public static void main(String[] args) throws Exception {
		// 省略上述代码
		for (Bean bean : beanList) {
			if("org.lqk.container.bean.BeanB".equals(bean.getClassName())){
				// 加载类并创建对象
				Class clazz = Class.forName("org.lqk.container.bean.BeanB");
				Object obj = clazz.newInstance();
				// 为对象属性赋值
				List<Property> props = bean.getProps();
				String methodName = "set" + bigCaseFirstChar(props.get(0).getName());
				Method m = clazz.getMethod(methodName, String.class);
				m.invoke(obj, props.get(0).getValue());
				// 使用对象
				BeanB beanB = (BeanB)obj;
				System.out.println(beanB.getTitle());
				break;
			}
		}
	}
	// 将该字符串的首字母大写
    public static String bigCaseFirstChar(String prop) {
		char ch = (char) (prop.charAt(0) - 32);
		return prop.replaceFirst(prop.charAt(0) + "", ch + "");
    }
}

我们真的弄了一个类出来,虽然对经常用spring的人来说有点大惊小怪,但亲身体验一把,还是很有意义的。

posted @ 2015-11-05 22:25  骨汤鸡蛋面  阅读(301)  评论(0编辑  收藏  举报