自己构建一个Spring自定义标签以及原理讲解

    平时不论是在Spring配置文件中引入其他中间件(比如dubbo),还是使用切面时,都会用到自定义标签。那么配置文件中的自定义标签是如何发挥作用的,或者说程序是如何通过你添加的自定义标签实现相应的功能的呢?且看下文。

通过对本文的阅读,你会在阅读涉及到自定义标签的源码功能时事半功倍,而且还可以自己动手做出一个自己的自定义标签。

     先呈上我自己在本地实现自定义标签的代码及对应讲解:

1、先无脑输出一个测试要用到的Bean类

 1 public class User {
 2 
 3     private String userName;
 4     private String emailAddress;
 5 
 6     public String getUserName() {
 7         return userName;
 8     }
 9 
10     public void setUserName(String userName) {
11         this.userName = userName;
12     }
13 
14     public String getEmailAddress() {
15         return emailAddress;
16     }
17 
18     public void setEmailAddress(String emailAddress) {
19         this.emailAddress = emailAddress;
20     }
21 }
View Code

 2、spring的xml配置文件,以及在配置文件中引入自定义标签跟它的命名空间

 1 <?xml version="1.0" encoding="UTF-8"?>
 2 <beans xmlns="http://www.springframework.org/schema/beans"
 3        xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
 4        xmlns:context="http://www.springframework.org/schema/context"
 5        xmlns:aop="http://www.springframework.org/schema/aop"
 6        xmlns:mvc="http://www.springframework.org/schema/mvc"
 7        xmlns:myname="http://www.zzq.com/schema/user"
 8        xsi:schemaLocation="http://www.springframework.org/schema/beans
 9       http://www.springframework.org/schema/beans/spring-beans-4.0.xsd
10       http://www.springframework.org/schema/context
11       http://www.springframework.org/schema/context/spring-context-4.0.xsd
12       http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop.xsd
13       http://www.springframework.org/schema/mvc http://www.springframework.org/schema/mvc/spring-mvc.xsd
14       http://www.zzq.com/schema/user http://www.zzq.com/schema/user.xsd">
15 
16     <aop:aspectj-autoproxy proxy-target-class="true"/>
17     <mvc:annotation-driven/>
18     <context:component-scan base-package="myDemoHome"/>
19     <context:property-placeholder location="classpath:properties/config.properties" ignore-unresolvable="true"/>
20 
21     <myname:myPw id="testUserBean" userName="zzq" emailAddress="zzqing@110.com"/>
22 
23 </beans>
View Code

3、从2中可以看到,命名空间中我添加了自定义的xmlns:myname="http://www.zzq.com/schema/user",以及http://www.zzq.com/schema/user跟http://www.zzq.com/schema/user.xsd。

其中紧跟xmlns冒号后面的部分,就是我们自定义标签引号前的部分,比如此处定义了myname,那么自定义标签中我就可以<myname:XXX/>这样引用了,其中的XXX则是在命名空间中定义的myPw。

中间http://www.zzq.com/schema/user对应此自定义标签的handler,放在Spring.handlers中。

最后的http://www.zzq.com/schema/user.xsd则定义了此自定义标签的XXX,即自定义标签冒号后面有什么,由此xsd定义,放在Spring.schemas中。

Spring.handlers跟Spring.schemas文件都放在META-INF目录下,因为spring会默认去此目录下读。

Spring.handlers如下所示:

http\://www.zzq.com/schema/user=myDemoHome.springElement.bdParser.UserNamespaceHandler

Spring.schemas如下所示:

http\://www.zzq.com/schema/user.xsd=META-INF/spring-test.xsd

3.1  自定义标签的解析类UserNamespaceHandler构建

 1 package myDemoHome.springElement.bdParser;
 2 
 3 import myDemoHome.springElement.User;
 4 import org.springframework.beans.factory.support.BeanDefinitionBuilder;
 5 import org.springframework.beans.factory.xml.AbstractSingleBeanDefinitionParser;
 6 import org.springframework.util.StringUtils;
 7 import org.w3c.dom.Element;
 8 
 9 public class UserBeanDefinitionParser extends AbstractSingleBeanDefinitionParser {
10 
11     @Override
12     protected Class getBeanClass(Element element) {
13         return User.class;
14     }
15 
16     @Override
17     protected void doParse(Element element, BeanDefinitionBuilder builder) {
18         String name = element.getAttribute("userName");
19         String address = element.getAttribute("emailAddress");
20 
21         if (StringUtils.hasText(name)) {
22             builder.addPropertyValue("userName", name);
23         }
24         if (StringUtils.hasText(address)) {
25             builder.addPropertyValue("emailAddress", address);
26         }
27     }
28 
29 }
View Code

就是对element中的标签进行解析处理,完成从xml中的标签属性向对象值的转化

3.2 自定义标签解析类的注册 UserNamespaceHandler

 1 package myDemoHome.springElement.bdParser;
 2 
 3 import org.springframework.beans.factory.xml.NamespaceHandlerSupport;
 4 
 5 public class UserNamespaceHandler extends NamespaceHandlerSupport {
 6     @Override
 7     public void init() {
 8         registerBeanDefinitionParser("myPw", new UserBeanDefinitionParser());
 9     }
10 }
View Code

此处的意思就是当遇到myPw这个标签的时候,往spring容器中注入这个标签的解析类,以完成后续对标签属性的解析。看到此处,各位道友有没有想起AOP的自定义注解aspectj-autoproxy 的解析呢?其实套路都是一样的。

3.3 xsd文件spring-test.xsd的定义

<?xml version="1.0" encoding="UTF-8" ?>
<schema xmlns="http://www.w3.org/2001/XMLSchema"
        targetNamespace="http://www.zzq.com/schema/user"
        elementFormDefault="qualified">

    <element name="myPw">
        <complexType>
            <attribute name="id" type="string"/>
            <attribute name="userName" type="string"/>
            <attribute name="emailAddress" type="string"/>
        </complexType>
    </element>
</schema>
View Code

此文件规定了自定义注解的标签,以及对应的属性

4、测试类ElementTest

 1 package myDemoHome.springElement;
 2 
 3 import org.springframework.context.ApplicationContext;
 4 import org.springframework.context.support.ClassPathXmlApplicationContext;
 5 
 6 public class ElementTest {
 7     public static void main(String[] args) {
 8         ApplicationContext applicationContext = new ClassPathXmlApplicationContext("classpath*:spring/spring-config.xml");
 9         User user = (User)applicationContext.getBean("testUserBean");
10         System.out.println(user.getEmailAddress());
11     }
12 }
View Code

再附上一张所用类的位置关系图

至此大功告成,最后运行一下测试类:

功德圆满!

总结:

构建一个自定义标签的流程便是如此,相信如果后面再遇到自定义标签,按照此构建思路反向解析一下便也能顺藤摸瓜知晓它的来龙去脉。其实看一下之前我们用过的中间件,像dubbo,也是一样的套路,只是功能更繁杂。

dubbo的jar包如图所示:

也是用了这三个实现的标签引用。

写在最后:至此,19年金三银四找工作之途的总结(包括技术跟个人感悟,感兴趣的道友可以移步我的另几篇博文一探究竟)便告一段落。今年自从去年下半年的资本收缩后,行情确实相较以往差了一些(虽然贫道也才17年入行),但我个人的感觉

是行业整体回归理性,之前是经过三四个月培训班下来就月薪十几K而且都抢着招人,太不正常。话说回来,不管外界行情怎样,只要你喜欢做这一行,都要静下心来多学习多探究,才能在不断的积累中让自己更上一层。今年年初两次阿里的面试

经历也让自己对于自身能力有了更客观真实的认识,不管是从项目经历还是个人技术积累上,都差很多。目前贫道新公司所处的部门,对技术比较注重,且技术大佬不少,工作都很认真,正是"发粪涂墙"之际,内心充满激情。后面的博文目前打算

更多偏向微服务(因为目前公司的项目用的就是微服务架构)、并发编程(因为前段时间买了本并发编程实战)。技术积累不是一朝一夕,与各位道友共勉!

 

posted on 2019-04-21 12:10  淡墨痕  阅读(2199)  评论(1编辑  收藏  举报