spring自定义标签之 自我实现
引言: 最近心情比较难以平静,周末的两天就跑出去散心了,西湖边上走走,看日落,还是不错的。回来博客上发现,在自定义标签上,最后一步实现忘记加上了。其实,人生的路程中,我们总是实现着自我的价值,让自己的生活更有意义。
在标签的定义完,也只是自我实现的一半,对于按我们的要求所定义的配置信息,自然而然的需要为这些定义各个属性进行解析和进一步的操作处理了。
进一步问题: 对于前一篇(spring自定义标签之二 —— 规范定义XSD )定义下来的xml的标签定义,如何对其进行解析的问题了。
自定义的标签如下:
- <mysql:client id="sqlMapClient" datasouceip="localhsost" characterEncoding="utf8"
- dbname="freebug" username="root" password="root"
- configLocation="classpath:SqlMapCommonConfig.xml" />
具体实现:
对于在spring的配置文件中已经进行了声明标签,这些可以上(上一节的规范定义已经说明了)。在上一节中也提到了,需要在资源文件中加入几个文件。

其中springtag.xsd及spring.schemas是为标签定义使用的,而spring.handlers是为了进行声明解释实handler现使用的。
在解析自定义的标签时,对于基本简单的自定义标签可以使用如下方式。继承,两个基类,进行实现。

图1. 实现自定义标签的实现类图
被继承的基类,为spring中带有的基类:
1、NamespaceHandlerSupport
2、AbstractSimpleBeanDefinitionParser
实现类为:
1、TagsNamespaceHandler
- package config;
- import org.springframework.beans.factory.xml.NamespaceHandlerSupport;
- /**
- * 注册定自义标签对应的解析类
- *
- * @author sammor
- * @date 2011-6-27 上午10:52:44
- */
- public class TagsNamespaceHandler extends NamespaceHandlerSupport {
- @Override
- public void init() {
- //自定义标签中的element标签名为client解析注册使用MysqlMapClientPraser进行.
- registerBeanDefinitionParser("client", new MysqlMapClientPraser());
- }
- }
2、MysqlMapClientPraser
- package config;
- import org.springframework.beans.factory.support.BeanDefinitionBuilder;
- import org.springframework.beans.factory.xml.AbstractSimpleBeanDefinitionParser;
- import org.springframework.beans.factory.xml.ParserContext;
- import org.springframework.core.io.ClassPathResource;
- import org.springframework.jdbc.datasource.DriverManagerDataSource;
- import org.springframework.orm.ibatis.SqlMapClientFactoryBean;
- import org.springframework.orm.ibatis.SqlMapClientTemplate;
- import org.w3c.dom.Element;
- /**
- * 标签解析处理
- *
- * @author sammor
- * @date 2011-6-27
- */
- public class MysqlMapClientPraser extends AbstractSimpleBeanDefinitionParser {
- /**
- * element 相当于对应的element元素 parserContext 解析的上下文 builder 用于该标签的实现
- */
- @Override
- protected void doParse(Element element, ParserContext parserContext,
- BeanDefinitionBuilder builder) {
- // 从标签中取出对应的属性值
- String dbname = element.getAttribute("dbname");
- String datasouceip = element.getAttribute("datasouceip");
- String username = element.getAttribute("username");
- String password = element.getAttribute("password");
- String characterEncoding = element.getAttribute("characterEncoding");
- String configLocation = element.getAttribute("configLocation");
- final String driverClassName = "com.mysql.jdbc.Driver";
- // System.out.println("dbname" + dbname);
- // System.out.println("datasouceip" + datasouceip);
- // System.out.println("username" + username);
- // System.out.println("password" + password);
- // System.out.println("characterEncoding" + characterEncoding);
- // System.out.println("configLocation" + configLocation);
- final StringBuffer url = new StringBuffer("jdbc:mysql://");
- url.append(datasouceip).append("/").append(dbname).append(
- "?useUnicode=true").append("&").append(
- "characterEncoding=" + characterEncoding).append(
- "&autoReconnect=true");
- // 创建 datasource实例
- DriverManagerDataSource datasource = new DriverManagerDataSource();
- datasource.setDriverClassName(driverClassName);
- // System.out.println(url.toString());
- datasource.setUrl(url.toString());
- datasource.setUsername(username);
- datasource.setPassword(password);
- // 创建SqlMapClientFactoryBean实例
- SqlMapClientFactoryBean sqlmapclient = new SqlMapClientFactoryBean();
- sqlmapclient.setDataSource(datasource);
- sqlmapclient.setConfigLocation(new ClassPathResource(configLocation));
- try {
- sqlmapclient.afterPropertiesSet();
- } catch (Exception e) {
- parserContext.getReaderContext().error(
- "sqlmapclient.afterPropertiesSet error", e);
- }
- // 把创建完的实例对应的传到该标签类实现的相应属性中
- builder.addPropertyValue("dataSource", datasource);
- builder.addPropertyValue("sqlMapClient", sqlmapclient.getObject());
- ;
- }
- @Override
- protected Class getBeanClass(Element element) {
- // 返回该标签所定义的类实现,在这里是为了创建出SqlMapClientTemplate对象
- return SqlMapClientTemplate.class;
- }
- }
对标签的实现类写完之后,需要声明该handler。通过spring.handlers 文件进行声明:
- http\://sammor.javaeye.com/schema/tags=config.TagsNamespaceHandler
测试环节:
配置完成,进行测试。
1、spring配置文件填写配置信息
- <mysql:client id="sqlMapClientTemplate" datasouceip="localhost"
- dbname="freebug" characterEncoding="utf8" username="root" password="root"
- configLocation="SqlMapCommonConfig.xml" />
- <bean id="usersinfoDAO" class="com.dbms.dao.UsersinfoDAOImpl">
- <property name="sqlMapClientTemplate" ref="sqlMapClientTemplate"></property>
- </bean>
2、单元测试
- ApplicationContext ac = new ClassPathXmlApplicationContext(
- "classpath:applicationContext.xml");
- UsersinfoDAO user = (UsersinfoDAO) ac.getBean("usersinfoDAO");
- System.out.println("记录数:" + user.selectByExample(null).size());
3、测试结果:
- 记录数:6
结论
个人觉得自定义标签的应用可以很广,但如何去利用好这个便利才是一个问题,并不是把什么都自定义化才是最好的。自定义标签的目的是为了更好的方便我们的开发,对一些繁琐而又固定的东西,进行一次的封装配置化以减少问题等实现其价值的自我实现。
spring自定义标签之 自我实现的更多相关文章
- spring自定义标签之  规范定义XSD
		引言: spring的配置文件中,一切的标签都是spring定义好的.<bean/>等等,有了定义的规范,才能让用户填写的正常可用.想写自定义标签,但首先需要了解XML Schema De ... 
- spring基础---->spring自定义标签(一)
		Spring具有一个基于架构的扩展机制,可以使用xml文件定义和配置bean.本博客将介绍如何编写自定义XML bean的解析器,并用实例来加以说明.其实我一直相信 等你出现的时候我就知道是你. Sp ... 
- Spring 自定义标签配置
		前景:经常使用一些依赖于Spring的组件时,发现可以通过自定义配置Spring的标签来实现插件的注入,例如数据库源的配置,Mybatis的配置等.那么这些Spring标签是如何自定义配置的?学习Sp ... 
- Spring自定义标签
		一.原理: 1.Spring通过XML解析程序将其解析为DOM树, 2.通过NamespaceHandler指定对应的Namespace的BeanDefinitionParser将其转换成BeanDe ... 
- 自己构建一个Spring自定义标签以及原理讲解
		平时不论是在Spring配置文件中引入其他中间件(比如dubbo),还是使用切面时,都会用到自定义标签.那么配置文件中的自定义标签是如何发挥作用的,或者说程序是如何通过你添加的自定义标签实现相应的功能 ... 
- spring 自定义标签的实现
		在我们进行Spring 框架开发中,估计用到最多的就是bean 标签吧,其实在Spring中像<mvc/><context/>这类标签以及在dubbo配置的标签都是属于自定义的 ... 
- Spring自定义标签解析与实现
		在Spring Bean注册解析(一)和Spring Bean注册解析(二)中我们讲到,Spring在解析xml文件中的标签的时候会区分当前的标签是四种基本标签(import.alias ... 
- spring自定义标签学习
		看到几篇很全的自定义标签,从定义到使用,写的很好. 这里我也是在那里学习的,对学习spring源码也很有帮助. 贴出来与大家共享. http://sammor.iteye.com/blog/11009 ... 
- Spring自定义标签的实现
		首先 简单写下 spring xml解析的过程 通过一段简单的 调用spring代码开始 public static void main(String[] args) { ApplicationCon ... 
随机推荐
- Transaction And Lock--已提交读快照
			--===================================================== --行版本控制已提交读ALTER DATABASE DB5 SET READ_COMMI ... 
- Partition--分区切换
			现有数据表[dbo].[staging_TB1_20131018-104722]和分区表[dbo].[TB1],需要将分区表和数据表中做数据交换 CREATE TABLE [dbo].[staging ... 
- c# 类的反射实例 (GetType().Invoke().GetMethod().CreateInstance())
			原文:http://www.cnblogs.com/chenwei19/archive/2009/02/04/1384034.html Class1和Form 窗体在同一个命名空间 using Sys ... 
- 基于.net standard 的动态编译实现
			在前文[基于.net core 微服务的另类实现]结尾处,提到了如何方便自动的生成微服务的客户端代理,使对于调用方透明,同时将枯燥的东西使用框架集成,以提高使用便捷性.在尝试了基于 Emit 中间语言 ... 
- tf-idf  sklearn
			第一步:语料转化为词袋向量 step 1. 声明一个向量化工具vectorizer: 本文使用的是CountVectorizer,默认情况下,CountVectorizer仅统计长度超过两个字符的词, ... 
- OCP 12c最新考试原题及答案(071-8)
			8.(5-4) choose the best answer:You need to produce a report where each customer's credit limit has b ... 
- “全栈2019”Java第七十四章:内部类与静态内部类相互嵌套
			难度 初级 学习时间 10分钟 适合人群 零基础 开发语言 Java 开发环境 JDK v11 IntelliJ IDEA v2018.3 文章原文链接 "全栈2019"Java第 ... 
- TCP/IP学习笔记(2)-数据链路层
			数据链路层有三个目的: 为IP模块发送和接收IP数据报. 为ARP模块发送ARP请求和接收ARP应答. 为RARP发送RARP请求和接收RARP应答 ip大家都听说过.至于ARP和RARP,ARP叫做 ... 
- Python strip()函数用法
			Python中字符串处理函数里有三个去空格(包括'\n', '\r', '\t', ' ')的函数: strip 同时去掉左右两边的空格lstrip 去掉左边的空格rstrip 去掉右边的空格 具体示 ... 
- shell 常用命令集合
			grep -i 忽略大小写 -I 跳过二进制文件 -c 计算数量 -n 显示行号 -R 递归 -v 不匹配某个关键字 常用组合命令 grep -iIRn keyword * 搜索含有该 keyword ... 
