自定义标签在aop、事务标签等都有使用,因此值得好好分析一下。
1、代码分析
自定义的标签如下:
<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 文件进行声明:
Xml代码
http\://sammor.javaeye.com/schema/tags=config.TagsNamespaceHandler
2、测试环节:
配置完成,进行测试。
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
结论
个人觉得自定义标签的应用可以很广,但如何去利用好这个便利才是一个问题,并不是把什么都自定义化才是最好的。自定义标签的目的是为了更好的方便我们的开发,对一些繁琐而又固定的东西,进行一次的封装配置化以减少问题等实现其价值的自我实现。