自己动手写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的人来说有点大惊小怪,但亲身体验一把,还是很有意义的。

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

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值