zoukankan      html  css  js  c++  java
  • [Java] 解决spring的xml标签内不能自由增加说明的难题,方便调试、部署时进行批量屏蔽

    作者:zyl910

      以往我们想在spring的xml配置文件中增加说明文本时,只能使用xml注释(<!-- 注释 -->)。这对于“调试、部署时想批量屏蔽部分bean”是不利的。于是本文讨论如何解决这个难题,并给出一个对项目配置改动少的方案。

    一、最初问题

      例如现在想用quartz管理一个定时器类,故需要先在spring的xml配置文件中把该定时器类(JobTest)配置为bean。一般是这样写——

    <!-- 测试Job. -->
    <bean id="jobTest" class="org.zyl910.zyllibj.test.spring.JobTest" />
    

      上面就是利用了xml注释,来存储该bean的说明。虽然清晰,但它占用了xml注释。

      xml规定注释不能嵌套。于是若想批量屏蔽一些bean时,得细心跳过原来的xml注释说明部分,单个单个给注释掉。不但要花很多功夫,且一不小心容易造成xml注释不匹配,整个文件失效。

      故考虑不再使用xml注释来填写说明文本。

    二、最初问题的解决办法

      首先尝过在bean标签内直接增加一个特性(attribute)来填写说明,例如——

    <bean id="jobTest" class="org.zyl910.zyllibj.test.spring.JobTest" title="测试Job" />
    

      但是该办法是行不通的。因spring会检查xml标签内的特性,若发现其他特性便会报错退出。于是我们只能另外想办法。

      想到这个bean配的是我们自己的类,故我们可以在自己的类里加一个“title”属性(property),然后在xml里配置该bean的属性。例如——

    <bean id="jobTest" class="org.zyl910.zyllibj.test.spring.JobTest">
    	<property name="title" value="测试Job"/>
    </bean>
    

    三、真正难题

      用quartz管理定时器时,除了需配置类的bean外,还需要配置方法、触发器的bean,最后还要配置调度器。例如——

    <bean id="jobTest" class="org.zyl910.zyllibj.test.spring.JobTest">
    	<property name="title" value="测试Job"/>
    </bean>
    <bean id="jobTest_process" class="org.springframework.scheduling.quartz.MethodInvokingJobDetailFactoryBean">
    	<property name="targetObject" ref="jobTest"/>
    	<property name="targetMethod"><value>process</value></property>
    	<property name="concurrent" value="false"/>
    </bean>
    <bean id="jobTest_processTrigger" class="org.springframework.scheduling.quartz.CronTriggerFactoryBean">
    	<property name="jobDetail" ref="jobTest_process"/>
    	<property name="cronExpression" value="* 0/1 * * * ?"/>
    </bean>
    
    <bean class="org.springframework.scheduling.quartz.SchedulerFactoryBean">
    	<property name="triggers">
    		<list>
    			<ref bean="jobTest_processTrigger"/>
    		</list>
    	</property>
    </bean>
    

      现在就又存在这样问题——想为“triggers”里的项目增加说明该怎么办?

      因为根据调试、部署需要,triggers里的定时器是经常需要使用xml注释来批量控制启停的。如果用xml注释写说明,就没法批量启停了。

      其次,先前也试过,spring标签里不能加特性(attribute)。

      第三,这里用的是ref标签,故不能像先前对bean标签那样加个属性(property)来解决.

      第四,这里是在list内加定时器,故不能用其他spring标签放其他类型的数据.

      在网上搜索了很久,发现spring支持xml扩展,但该方案不适合我们项目。因为那种方法对项目配置改动较多。而对于大型项目来说,是拆分为许多个子项目,若用spring扩展的话影响太大了,其他子项目不一定同意这样的更改,没法形成统一方案。

    四、解决方案

      我对着spring的官方文档、源码看了很久,最终想出来了一套简单有效的解决方案。

      新建一个“TitleProxyFactoryBean”类,它继承自spring的“ProxyFactoryBean”类,并提供title等属性。代码如下——

    package org.zyl910.zyllibj.spring;
    
    import org.springframework.aop.framework.ProxyFactoryBean;
    
    /** 带标题的代理Bean工厂.
     * 
     * <p>由于spring的xml里不能直接加特性写注释,且spring的xml扩展机制对项目改动较大. 故制定了此类来实现为bean加说明的功能.</p>
     * 
     * 提供了3个可选属性:
     * <ul>
     *   <li>String title: 标题. 可用于介绍该bean是什么.</li>
     *   <li>String group: 组. 有时需要对bean进行分组,此时可把分组说明填这里.</li>
     *   <li>Object tag: 标记. 可用根据需要填充任意内容.</li>
     * </ul>
     * 
     * @author zyl910
     *
     */
    public class TitleProxyFactoryBean extends ProxyFactoryBean {
    	private static final long serialVersionUID = -4213438026061725669L;
    	
    	/** 标题. 可用于介绍该bean是什么. */
    	private String title;
    	/** 组. 有时需要对bean进行分组,此时可把分组说明填这里. */
    	private String group;
    	/** 标记. 可用根据需要填充任意内容. */
    	private Object tag;
    	
    	/** 取得标题.
    	 * @return the title
    	 */
    	public String getTitle() {
    		return title;
    	}
    	
    	/** 设置标题.
    	 * @param title the title to set
    	 */
    	public void setTitle(String title) {
    		this.title = title;
    	}
    	
    	/** 取得组.
    	 * @return the group
    	 */
    	public String getGroup() {
    		return group;
    	}
    	
    	/** 设置组.
    	 * @param group the group to set
    	 */
    	public void setGroup(String group) {
    		this.group = group;
    	}
    	
    	/** 取得标记.
    	 * @return the tag
    	 */
    	public Object getTag() {
    		return tag;
    	}
    	
    	/** 设置标记.
    	 * @param tag the tag to set
    	 */
    	public void setTag(Object tag) {
    		this.tag = tag;
    	}
    
    }
    

      然后在配置文件中便可以用代理的方式引用bean,并可填写说明。例如——

    <bean class="org.zyl910.zyllibj.spring.TitleProxyFactoryBean"><property name="target" ref="jobTest_processTrigger"/><property name="title" value="测试Job"/></bean>
    

      测试通过。

    五、配置优化

      上面的方案虽然能解决问题,但是太长了,不易读。

      此时可做2点改进——

    1. 使用spring的p来配置,它比property标签精简很多。
    2. 将重要属性放在前面,而class等无需改的放后面。

      例如——

    <bean p:target-ref="jobTest_processTrigger" p:title="测试Job" class="org.zyl910.zyllibj.spring.TitleProxyFactoryBean" />
    

      优化后测试通过。

    (完)

    源码地址: https://coding.net/u/zyl910/p/zyllibj/git

  • 相关阅读:
    《Cracking the Coding Interview》——第7章:数学和概率论——题目4
    《Cracking the Coding Interview》——第7章:数学和概率论——题目3
    《Cracking the Coding Interview》——第7章:数学和概率论——题目2
    最小二乘拟合
    设置手机邮件下载文件路径
    #pragma data_seg() 共享数据// MyData段 // 进程 // DLL
    树状数组板子 x
    博弈论 x
    luogu P1147 连续自然数和 x
    luogu P1068 分数线划定 x
  • 原文地址:https://www.cnblogs.com/zyl910/p/zyllibj_TitleProxyFactoryBean.html
Copyright © 2011-2022 走看看