这篇随笔的许多知识来源于:http://www.importnew.com/19391.html

之所以会去看这些东东,主要是希望能够模仿spring mvc的处理流程,做出一套合理的交易处理流程。

之前已经根据网上查到的知识,做了一些尝试,只要按照如下流程,就可以使用自定义的命名空间:

1.通过在spring的配置文件中配置自己的命名空间

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:sb="http://www.springbank.com/schema/springbank"
xsi:schemaLocation="
http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-2.0.xsd
http://www.springbank.com/schema/springbank http://www.springbank.com/schema/springbank.xsd"> <!-- 新建客户 -->
<sb:transaction id="CreateOwner" action="CreateOwnerAction" template="TransTemplate"></sb:transaction>
</beans>

2.可以看到上面的xsd文件是一个web地址,我现在自然没有这个东东,所以要在spring.schemas这个文件中映射本地位置,而这个文件必须要放在CLASSPATH:META-INF下面

http\://www.springbank.com/schema/springbank.xsd=com/springbank/test/example/mytag/springbank.xsd

但实际上可能是IDE比较傻,每次validate都会说找不到,但实际上运行起来的时候是能证明它是被找到的

3.需要向spring注册我们的标签,和标签解析器,我们的标签。固定的套路,代码如下:

public class SpringbankNamespaceHandlerSupport extends NamespaceHandlerSupport{
private final Logger log = LoggerFactory.getLogger(getClass());
@Override
public void init() {
registerBeanDefinitionParser("transaction", new TagTransactionBeanDefinitionParser());
}
}

其中TagTransactionBeanDefinitionParser如下

public class TagTransactionBeanDefinitionParser extends AbstractSingleBeanDefinitionParser {
private final Logger log = LoggerFactory.getLogger(getClass()); @Override
protected Class<?> getBeanClass(Element element) {
return Transaction.class;
}
protected void doParse(Element element, BeanDefinitionBuilder bean) {
String id = element.getAttribute("id");
String action = element.getAttribute("action");
String template = element.getAttribute("template"); if (StringUtils.hasText(id)) {
bean.addPropertyValue("id", id);
}
if (StringUtils.hasText(action)) {
bean.addPropertyValue("action", action);
}
if (StringUtils.hasText(template)) {
bean.addPropertyValue("template", template);
}
}
}

最后是我的测试类:

public static void main(String[] args) {
ApplicationContext context = new ClassPathXmlApplicationContext("com/springbank/test/example/mytag/applicationContext.xml");
Transaction transaction = (Transaction)context.getBean("CreateOwner");
System.out.println(transaction.getTemplate());
}

输出:

Jul 11, 2017 3:57:22 PM org.springframework.context.support.ClassPathXmlApplicationContext prepareRefresh
INFO: Refreshing org.springframework.context.support.ClassPathXmlApplicationContext@49c2faae: startup date [Tue Jul 11 15:57:22 CST 2017]; root of context hierarchy
Jul 11, 2017 3:57:22 PM org.springframework.beans.factory.xml.XmlBeanDefinitionReader loadBeanDefinitions
INFO: Loading XML bean definitions from class path resource [com/springbank/test/example/mytag/applicationContext.xml]
Bean Transaction have been created: Transaction [id=null, template=null]
TransTemplate

做法就是这样的做法,但是如果我要搞一个<sb:annotation-driven>,又改怎么办呢?所以只知道做法就没有办法举一反三,必须要知道原理,所以我先来研究上面的原理,主要是为什么我自己定义的SpringbankNamespaceHandlerSupport 能够工作

我是new了一个ClassPathXmlApplicationContext,然后看它一步一步怎么处理。

在BeanDefinitionParserDelegate类中:

public BeanDefinition parseCustomElement(Element ele, BeanDefinition containingBd) {
String namespaceUri = getNamespaceURI(ele);
NamespaceHandler handler = this.readerContext.getNamespaceHandlerResolver().resolve(namespaceUri);
if (handler == null) {
error("Unable to locate Spring NamespaceHandler for XML schema namespace [" + namespaceUri + "]", ele);
return null;
}
return handler.parse(ele, new ParserContext(this.readerContext, this, containingBd));
}

经过调试可以看到:

另一方面我发现在

org.springframework.context.annotation.ConfigurationClassParser

里面有doProcessConfigurationClass这个方法,对一些注解进行了了处理。这里我还没有看懂

另一方面,根据开头那边文章里的知识在ComponentScanBeanDefinitionParser类里的parse方法对

<context:component-scan base-package="com.xxx" />

进行了处理

所以我有了一个想法,就是我搞的<sb:annotation-dvriven/>我也搞这样一个parser去处理,扫描指定的包下面的信息,然后放到一个list中,当请求来的时候,根据交易类型,去遍历这个list

springbank 开发日志 Spring启动过程中对自定义标签的处理的更多相关文章

  1. 死磕Spring之IoC篇 - 解析自定义标签(XML 文件)

    该系列文章是本人在学习 Spring 的过程中总结下来的,里面涉及到相关源码,可能对读者不太友好,请结合我的源码注释 Spring 源码分析 GitHub 地址 进行阅读 Spring 版本:5.1. ...

  2. Windows平台下Oracle实例启动过程中日志输出

    Windows平台下Oracle实例启动过程中日志输出记录. 路径:D:\app\Administrator\diag\rdbms\orcl\orcl\trace\alert_orcl.log 输出内 ...

  3. Windows平台下Oracle监听服务启动过程中日志输出

    Windows平台下Oracle监听服务启动过程中日志输出记录. 日志目录:D:\app\Administrator\diag\tnslsnr\WIN-RU03CB21QGA\listener\tra ...

  4. 嵌入式Linux启动过程中的问题积累

    嵌入式Linux启动过程中的问题积累 Dongas 07-12-19 1.Bad Magic Number ## Booting image at 33000000 ... Bad Magic Num ...

  5. 【ARM-Linux开发】U-Boot启动过程--详细版的完全分析

    ---------------------------------------------------------------------------------------------------- ...

  6. hive配置以及在启动过程中出现的问题

    一.hive配置 1.安装环境 在hadoop-1.2.1集群上安装hive-1.2.1 2.将hive-1.2.1环境变量添加到PATH路径下 使用如下命令打开配置文件 nano /etc/prof ...

  7. 转:Tomcat启动过程中找不到JAVA_HOME JRE_HOME的解决方法

    转自:http://blog.sina.com.cn/s/blog_61c006ea0100l1u6.html 原文: 在XP上明明已经安装了JDK1.5并设置好了JAVA_HOME,可偏偏Tomca ...

  8. Tomcat启动过程中找不到JAVA_HOME解决方法

    在XP上明明已经安装了JDK1.5并设置好了JAVA_HOME,可偏偏Tomcat在启动过程中找不到. 报错信息如下:Neither the JAVA_HOME nor the JRE_HOME en ...

  9. 开发与测试整体过程中的Git分支merge流程

    开发与测试整体过程中的Git分支merge流程 Git分支merge之开发流程 首先在Gitlab上有个仓库存储着原始的项目代码,其中包含一个叫master的分支.然后可能按功能进行分配,由不同的开发 ...

随机推荐

  1. C# 优化程序的四十七种方法

    一.用属性代替可访问的字段 1..NET数据绑定只支持数据绑定,使用属性可以获得数据绑定的好处: 2.在属性的get和set访问器重可使用lock添加多线程的支持. 二.readonly(运行时常量) ...

  2. POJ - 1584 A Round Peg in a Ground Hole(判断凸多边形,点到线段距离,点在多边形内)

    http://poj.org/problem?id=1584 题意 按照顺时针或逆时针方向输入一个n边形的顶点坐标集,先判断这个n边形是否为凸包. 再给定一个圆形(圆心坐标和半径),判断这个圆是否完全 ...

  3. jmeter使用正则表达式匹配多个中的响应结果

    一.背景: 同一个正则表达式匹配多个响应结果值,之前都是添加多个正则表达式,一个一个去获取需要的值,比较麻烦:今天尝试了一下用一个正则表达式获取响应中所有需要的值,使用这种方式也能够获取多个结果中指定 ...

  4. 寻路优化(一)——二维地图上A*启发函数的设计探索

    工作中需要优化A*算法,研究了一天,最后取得了不错的效果.看网上的朋友还没有相关的研究,特此记录一下.有错误欢迎大家批评指正.如需转载请注明出处,http://www.cnblogs.com/Leon ...

  5. python 多线程中子线程和主线程相互通信

    主线程开启多个线程去干活,每个线程需要完成的时间不同,干完活以后都要通知给主线程,下面代码说明该应用: 代码块: import threading import queue import time i ...

  6. split('\r\n')

    '\r'是回车,'\n'是换行,前者使光标到行首,后者使光标下移一格.通常用的Enter是两个加起来. 实际我的脚本读取FTP的列表,如果用的split("\r\n"),可以获得正 ...

  7. Spring源码解读

    beanfactory https://www.cnblogs.com/lspz/p/6244948.html requestmapping https://blog.csdn.net/u012557 ...

  8. 深入剖析ThreadLocal实现原理以及内存泄漏问题

    关于ThreadLocalMap<ThreadLocal, Object>弱引用问题: 当线程没有结束,但是ThreadLocal已经被回收,则可能导致线程中存在ThreadLocalMa ...

  9. OKVIS 代码框架

    1. okvis_app_synchronous.cpp 在此文件中 okvis 对象为 okvis_estimator,是类 okvis::ThreadedKFVio 的实例化对象. 数据输入接口是 ...

  10. SpringBoot2.x过滤器Filter和使用Servlet3.0配置自定义Filter实战

    补充:SpringBoot启动日志 1.深入SpringBoot2.x过滤器Filter和使用Servlet3.0配置自定义Filter实战(核心知识) 简介:讲解SpringBoot里面Filter ...