java反编译能拿到源码吗_大牛带你解读Spring源码,编写自定义标签,您能学会吗?

Spring源码阅读之编写自定义标签

基于昨天的那篇文章,我们今天来聊一个比较简单轻松点的话题,今天我们来实现一个自定义的标签。

42c874f5d2496806647fa7d38eae33ae.png

(本文来自公众号:z小赵)
先明确下整个项目的结构,网上有很多关于自定义标签的实现方式,但是教程包结构不清晰,导致测试无法正常进行,博主自己也栽了一个坑,为了让朋友们可以快速验证结果,我们首先来介绍一下包结构。

44baebf64b8f5dc3436cb40da57ff4f6.png

第一个工程包结构,该工程的作用是为了实现一个自定义标签。

a95c35ed11718a6cba8003e8ef8b1da6.png

第二个工程是为了验证自定义标签是否能够正常工作。

这里说一下为什么创建了两个工程,博主就是在这里遇到了一个坑,最开始博主在一个工程下实现了标签并且去做验证,遇到了一个比较怪异的报错,如下,大概意思就是说 'mytag:annotation' 找不到,苦思冥想命名定义了为什么会找不到呢?朋友们混个眼熟,后面会讲为啥。

Caused by: org.xml.sax.SAXParseException; lineNumber: 10; columnNumber: 42; cvc-complex-type.2.4.c: 通配符的匹配很全面, 但无法找到元素 'mytag:annotation' 的声明。at com.sun.org.apache.xerces.internal.util.ErrorHandlerWrapper.createSAXParseException(ErrorHandlerWrapper.java:203)at

  1. 清楚了包结构,我们先来介绍第一个工程涉及文件的作用
  • MyNamespaceDefinitionParse:用于解析标签的具体实现
  • MyNamespaceHandler:命名空间处理器,用于注册标签对应的解析器
  • MyTagBean:自定义BeanClass
  • spring.handlers:指定使用的命名空间处理器,即指定MyNamespaceHandler
  • spring.schemas:用于指定.xsd文件位置的
  • resource-1.0.xsd:用于定义标签,也可理解成规范标签编写规则的
  1. 我们先来介绍下resource-1.0.xsd文件内容。
<?xml version="1.0" encoding="UTF-8"?>

  1. spring.schemas文件内容如下,用于指定resource-1.0.xsd文件的位置,左边为 key,右边为该文件在本地的位置,如果该文件在本地找不到会尝试去远程去找,即左边 key 指定的远程位置去获取。
http://www.zxz.com/resource/resource-1.0.xsd=/resource-1.0.xsd

  1. spring.handlers文件内容如下,用于指定自定义标签的解析器位置,左边为 key,右边为该文件在本地的位置。
http://www.zxz.com/resource=com.zxz.resource.MyNamespaceHandler

  1. MyTagBean.java文件,设置了 name myPackage两个属性,注意这个字段名和resource-1.0.xsd文件定义的名字实际上没有什么直接联系,只是名字相同而已。
package com.zxz.resource;/*** @author zxz*/public class MyTagBean {    private String name;    private String myPackage;    public String getName() {        return name;    }    public void setName(String name) {        this.name = name;    }    public String getMyPackage() {        return myPackage;    }    public void setMyPackage(String myPackage) {        this.myPackage = myPackage;    }    @Override    public String toString() {        return "MyTagBean{" + "name='" + name + ''' + ", myPackage='" + myPackage + ''' + '}';    }}

  1. MyNamespaceHandler.java文件内容如下,其继承了NamespaceHandlerSupport,实现了init方法。用于注册具体annotation 标签对应的解析器,这个和上次讲的的做法是一样的,不熟悉的朋友可以先看上一讲。
package com.zxz.resource;import org.springframework.beans.factory.xml.NamespaceHandlerSupport;/** * @author zxz */public class MyNamespaceHandler extends NamespaceHandlerSupport {    public void init() {        registerBeanDefinitionParser("annotation", new MyNamespaceDefinitionParse());    }}

  1. MyNamespaceDefinitionParse.java内容如下,其实现了BeanDefinitionParse接口,来实现parse方法。
package com.zxz.resource;import org.springframework.beans.MutablePropertyValues;import org.springframework.beans.factory.config.BeanDefinition;import org.springframework.beans.factory.support.RootBeanDefinition;import org.springframework.beans.factory.xml.BeanDefinitionParser;import org.springframework.beans.factory.xml.ParserContext;import org.w3c.dom.Element;/** * @author zxz */public class MyNamespaceDefinitionParse implements BeanDefinitionParser {    public BeanDefinition parse(Element element, ParserContext parserContext) {        RootBeanDefinition beanDefinition = new RootBeanDefinition();        // 设置beanClass        beanDefinition.setBeanClass(MyTagBean.class);        MutablePropertyValues mutablePropertyValues = beanDefinition.getPropertyValues();        // 添加name属性        if (element.hasAttribute("name")) {            mutablePropertyValues.addPropertyValue("name", element.getAttribute("name"));        }        // 添加package属性        if (element.hasAttribute("myPackage")) {            mutablePropertyValues.addPropertyValue("myPackage", element.getAttribute("myPackage"));        }        String id = element.getAttribute("id");        // 拿到注册表, 注册BeanDefinition        parserContext.getRegistry().registerBeanDefinition(id, beanDefinition);        return beanDefinition;    }}

至此,我们的自定义标签就制作完成了,接下来打成 jar 包,在第二个测试工程中进行测试。


  1. 第二个工程文件介绍。
  • applicationContext.xml:编写定义标签内容
  • Test.java:测试方法
  • pom.xml:引入第一个工程的 jar 包

  1. applicationContext.xml文件内容如下,在测试启动的时候会找 http://www.zxz.com/resource/resource-1.0.xsd和http://www.zxz.com/resource对应的.xsd文件和对应的处理器。回答下前面的那个报错原因,因为在 spring 启动的时候会先扫描本地所有的.handlers和.schemas文件,如果本地找不到回去远程找。但是从我们的报错可以看出,应该是没有扫描到我们自定义的两个文件,但是我们将其打成 jar 包,在另外一个工程里引用就可以扫到,所以我猜想 spring 启动只扫描在 pom 文件指定的那些包下文件。
<?xml version="1.0" encoding="UTF-8"?>

  1. Test.xml文件内容如下。
import org.springframework.context.ApplicationContext;import org.springframework.context.support.ClassPathXmlApplicationContext;public class Test {    public static void main(String[] args) {        ApplicationContext context = new ClassPathXmlApplicationContext("classpath:applicationContext.xml");        System.out.println(context.getBean("myTagBean"));    }}

11. 运行测试代码,最终输出结果如下。

dd944cdda6b02f4742a92411684ec682.png

总结

以上即为创建一个自定义标签的流程,全部代码都经过测试并得出预期结果。

具体的运行原理我在上一篇讲解标签的源码解析讲解过了,不熟悉的朋友先看看上一篇内容。至此,我们关于标签解析成BeanDefinition对象并将其注册到BeanFactory的全部流程就讲完了,涉及到的其他细节朋友大家自行了解,或者可以和我交流。

下一篇文章我们来讲讲下一个比较重要的流程invokeBeanFactoryPostProcessors,希望大家和我继续坚持下去,搞定 Spring 最核心的灵魂。

感谢大家支持,多多转发关注不迷路~~

723865a7d3494d7233c93630c46b06bc.png

感谢大家支持~~

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

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值