`

用Velocity进行配置文件信息的集中管理

阅读更多

Apache从他诞生的那天起就改变着我们的生活方式。而Velocity的出现也不是个例外。Velocity是一个基于java的模板引擎(template engine)。它允许任何人仅仅简单的使用模板语言(template language)来引用由java代码定义的对象。 当Velocity应用于web开发时,界面设计人员可以和java程序开发人员同步开发一个遵循MVC架构的web站点,也就是说,页面设计人员可以只关注页面的显示效果,而由java程序开发人员关注业务逻辑编码。Velocity将java代码从web页面中分离出来,这样为web站点的长期维护提供了便利,同时也为我们在JSP之外又提供了一种可选的方案。

曾几何时,还记得自己每次对JSP页面做一点小小的改动,就需要痛苦的几秒钟等待时间,好让JSP容器按部就班的把JSP页面先生成为Java文件,然后又将该Java文件编译成class文件,最后再将结果返回。然后又可能由于是自己的一时疏忽导致的一个小bug,使得你不得不再次去重复同样的工作。现在,当我们有了 Velocity 之后,页面的修改不再变成一件让人痛苦的事情。它可以让你彻底体验页面改动带来的飞速回报。这也是 Velocity 值得称道的地方之一:他拯救了我们的WEB程序员们。于是,你Google一下,随便就可以在网上找到成千上万的关于 Velocity 如何进行Web开发的相关文章。

Velocity之所以如此famous、如此优秀,是因为Velocity的能力远不止web表现层开发这个领域,Velocity的诞生也为开发界带出了另一个话题——模板化(至少是使得模板化更加简单易行),而web表现层的应用仅仅只是“模板化”话题中的一个分支而已,我今天想聊一聊关于 Velocity 为大多数程序员所不够关注但是又非常重要的应用——配置文件模板化。
今天的软件开发界是一个开源插件满天飞的世界。什么Hibernate.cfg.xml、Hibernate.hbm.xml、struts-config.xml、spring.xml、web.xml甚至是tomcat、JBoss的一些配置都需要我们在发布产品的时候一并进行配置。其实,一个通常的工程其配置文件远远不止这些,在工程应用中,我们为了提高系统的灵活性,在开发阶段会引入大量的配置文件,而这些配置文件都会由于客户需求的变化、工程运行环境的不同出现不同的差异。同时又与操作系统(windows、Linux)、数据库、网络环境的参数有密切的关系。常态下,需要正确配置这些信息必须首先是一个对系统非常熟悉的Senior Engneer。但是,事实上是我们不可能为了解决这些系统配置的信息去每次找几个Senior Engneer去处理这些得不偿失的事情。那么问题就来了,既然不是由Senior Engneer来处理这些事情,那么那些其他的Junior Engneer有没有足够的能力去处理这些事情。有什么办法可以在这两者之间做出权衡呢。

一种可行的解决方案就是将这些配置成可替换的参数,然后找一个文件对这些需要配置的数据进行集中管理,这样当我们需要修改某些配置信息时,我们只需要在该文件中集中处理然后执行一个简单的替换操作所有的事情就全部OK了。于是我们想到了Velocity,这个看起来处理模板数据本该是天性使然的家伙,能否给我们一个预料之中的惊喜呢。其实我们的问题就转移到了,Velocity是否有能力将我们的参数一一进行提换的问题了。

小实验马上展开:
根据提供的接口我们很快写出了上面的这段代码,其中createFile() 方法是参数替换的具体实施者。
其中用到的两个配置文件内容分别如下:

java 代码
  1. package org.danlley.util.samples;   
  2.   
  3. import java.io.BufferedWriter;   
  4. import java.io.FileInputStream;   
  5. import java.io.FileWriter;   
  6. import java.util.Enumeration;   
  7. import java.util.Properties;   
  8.   
  9. import org.apache.velocity.Template;   
  10. import org.apache.velocity.VelocityContext;   
  11. import org.apache.velocity.app.Velocity;   
  12.   
  13. public class VelocityFirstStep{   
  14.     public static void main(String[] args){   
  15.         String resourceFileName="first_step.properties";   
  16.         String destPath="./target/classes/";   
  17.         String destFileName="generator.properties";   
  18.         try{   
  19.             createFile(resourceFileName,destPath,destFileName);   
  20.         }catch(Exception e){   
  21.             e.printStackTrace();   
  22.         }   
  23.     }   
  24.   
  25.     /**  
  26.      * <p>comments : danlley 【伏羲狂徒】  
  27.      *              mailto: danlley@126.com  
  28.      *              Blog: http://danlley.iteye.com  
  29.      *                    http://danlley.cublog.cn  
  30.      *          </p>  
  31.      * author  danlley 【伏羲狂徒】  
  32.      * coding date  2007-7-24  
  33.      * @param resourceFileName  
  34.      * @param path  
  35.      * @param destFileName  
  36.      * @throws Exception  
  37.      */  
  38.     public static void createFile(String resourceFileName,String path,String destFileName) throws Exception{   
  39.         Properties p=new Properties();   
  40.         p.setProperty(Velocity.INPUT_ENCODING,"GBK");   
  41.         p.setProperty(Velocity.OUTPUT_ENCODING,"GBK");   
  42.         Velocity.init(p);   
  43.         Template template=Velocity.getTemplate(path+destFileName);   
  44.         VelocityContext context=new VelocityContext();   
  45.         p.load(new FileInputStream(path+resourceFileName));   
  46.         Enumeration enu=p.keys();   
  47.         String key,value;   
  48.         while(enu.hasMoreElements()){   
  49.             key=enu.nextElement().toString();   
  50.             value=p.get(key).toString();   
  51.             context.put(key,new String(value.getBytes("ISO-8859-1"),"GBK"));   
  52.         }   
  53.         BufferedWriter writer=new BufferedWriter(new FileWriter(path+destFileName+".txt"));   
  54.         template.merge(context,writer);   
  55.         writer.flush();   
  56.         writer.close();   
  57.     }   
  58. }   


 

first_step.properties内容如下:

username=伏羲狂徒
passport=welcome to danlley's home page
messages=hello velocity

generator.properties中我们暂时只配置一条模板数据:

第三方第三方第三个大股东范甘迪说个======$username======改动热管热个人股热管热管大商股份第三个
passport=$passport
messages=$messages

运行程序得到生成文件generator.properties.txt

 

永远都不要忘了,老外的东西对我们中国的汉字从来都是照顾不周的。因此,我们在写程序的时候永远都应该对我们的汉字怀着悲天悯人之心,多加照顾(谁让我们不争气呢)。Velocity也理当如此,打开Velocity的源码,看看他在处理模板时是怎样设置编码格式的,当然程序员都是比较“懒惰”的,所以你也可以下面的代码仅仅是看看就行了:

java 代码
  1. System.out.println(template.getEncoding());//--默认是ISO-8859-1  


但是如果你真的对Velocity的编码设置感兴趣org.apache.velocity.runtime.resource包下的Resource类可能会对你有一定的参考价值。


运行程序后得到的generator.properties.txt内容如下:


第三方第三方第三个大股东范甘迪说个======伏羲狂徒======改动热管热个人股热管热管大商股份第三个
passport=welcome to danlley's home page
messages=hello velocity


这样,一个简单的模板管理的雏形就已经呈现在我们的面前了。其中first_step.properties在这里可以作为一个集中管理者,而generator.properties则是代替换的参数模板文件,最后generator.properties.txt则是我们需要的“熊掌、鱼翅”。思路已经有了,实现的可行性也已经验证了。那要想把他真正用在工程中,剩下的就只是对其进行扩充了。

看似好像没什么问题了,但是好像又不是很完美。为什么!用起来不方便,如果是在Ant和Maven中,我可以把整个参数替换过程放置在某个测试类中,然后运行一下,但是保证这个测试类的运行要先于其他测试类的运行就是一个令人头疼的事情,而且,参数替换过程本来就不属于工程代码的一部分,他应该是一个独立的组件才对。所以我们就把它开发成一个组件。不要拖动滚动条,因为下面没了,呵呵呵呵。我打算在下一个专题中来专门阐述一下基于Velocity的Ant和Maven插件开发过程。关注我的博客吧,^_^

 

参考资料:
http://ahuaxuan.iteye.com/blog/71430
http://www.open-open.com/open2821.htm

分享到:
评论
1 楼 galaxystar 2007-07-27  
楼主正解, velocity做配置文件的动态渲染是再好不过了!

相关推荐

    velocity的简单demo

    Velocity通过简单的语法和强大的数据整合能力,使得开发者能够将注意力集中在数据处理上,而将页面展示逻辑交由Velocity处理。在Java Web开发中,Velocity常用于生成HTML页面、邮件内容以及其他需要动态生成文本的...

    基于Java的分布式数据层TDDL框架源码

    框架基于集中式配置,采用Java语言开发,实现了对JDBC数据源的优化管理,具备主备切换、读写分离、动态数据库配置等强大功能。 技术构成: - 主要编程语言:Java - 辅助技术:JavaScript, CSS - 文件构成:共计2467...

    大数据时代下的财务管理创新

    2. 加强财务管理信息系统建设:建立和完善财务管理信息系统,实现财务数据的集中管理,提高数据处理和分析的能力。 3. 注重人才培养和能力提升:加强对财务人员的大数据技术培训,提高他们的数据分析能力,同时引进...

    毕业设计-struts 2Bhibernate实现的教务选课信息管理系统.zip

    比如,教师信息、课程信息、学生选课记录等数据,都可以通过定义对应的实体类并配置映射文件,由Hibernate自动处理CRUD(创建、读取、更新、删除)操作。 **整合Struts 2与Hibernate** 在实际开发中,Struts 2与...

    nim-docset:Dash,Zeal或Velocity中Nim的文档集

    2. `info.plist` 文件:这是一个配置文件,用于定义docset的元数据,如名称、作者、版本和图标等。 3. `icon.png` 图片文件:docset的图标,显示在Dash、Zeal或Velocity中以识别Nim文档集。 4. 可能还包含其他构建...

    Spring_MVC详解学习

    为了提高配置的便捷性,Spring MVC框架支持在不同环境下使用不同的配置文件。例如,可以在开发环境中使用一组配置,而在生产环境中使用另一组配置。Spring框架还支持通过不同的属性文件来管理不同环境下的配置项,...

    Spring_MVC详解学习.pdf

    Spring MVC提供了一种集中配置的方式,例如使用PropertyPlaceholderConfigurer和ContextLoaderListener等机制,来管理不同部署环境下的配置文件,这样就无需更改源代码即可切换不同的环境配置。 配置管理是任何Web...

    springboot学习

    chapter2-1-1:配置文件详解:自定义属性、随机数、多环境配置等 chapter2-1-2:2.0 新特性(一):配置绑定全解析 chapter2-2-1:2.0 新特性(二):新增事件ApplicationStartedEvent Web开发 chapter3-1-1:构建一...

    Struts2与spring整合事例

    4. **简化配置**:整合后,可以减少Struts2配置文件中的Action配置,将更多配置移到Spring中,使得配置更加集中和清晰。 5. **增强可测试性**:由于依赖注入,Action类可以通过Mock对象进行单元测试,提高了代码质量...

    Struts2所需最新jar包

    2. **配置文件**:Struts2使用XML配置文件(struts.xml或struts-default.xml)来定义Action、结果类型、拦截器等。这些配置可以集中管理,也可以分散到各个Action类中。 3. **拦截器(Interceptors)**:拦截器是...

    SSH (hibernate +struts2 + spring)

    Struts2通过Action类和配置文件定义了请求与业务处理的对应关系,提供了强大的拦截器机制,可以方便地进行日志记录、权限控制、事务管理等操作。此外,Struts2支持多种视图技术,如JSP、FreeMarker、Velocity等,...

    Struts 2,Spring 2,Hibernate S2SH 整合 例子 源代码

    - **hibernate.cfg.xml**:Hibernate的配置文件,包含数据库连接信息。 - **实体类(Entity)**:与数据库表映射的Java类,通常使用Hibernate注解进行配置。 - **持久化层(DAO)**:数据访问对象,负责与数据库的...

    C# 代码生成 MyGenerator

    3. **配置文件**:`MyGenerator.exe.config`是C#应用程序的标准配置文件,其中包含了程序运行时的配置信息,如连接字符串、应用程序设置等。开发者可以通过修改这个文件来定制MyGenerator的行为。 4. **第三方库...

    struts

    4. **良好的错误处理**:Struts提供了统一的异常处理机制,便于开发者对各种错误进行集中管理和处理。 5. **国际化支持**:Struts内置了对多语言的支持,开发者可以通过配置文件轻松实现应用的国际化。 **Struts的...

    struts2对webwork的改进

    Struts2提供了更直观和简洁的API,如`@Action`、`@Result`等注解,允许开发者在Action类和方法上直接进行配置,减少了XML配置文件的工作量。 7. **插件体系**: Struts2引入了插件机制,允许开发者轻松地添加或...

    Struts2-2,2.1.1官方完整版jar包

    这个配置文件可以集中管理整个应用的设置,也可以通过注解方式来简化配置。 7. **插件系统**:Struts2拥有丰富的插件支持,如Tiles用于布局管理,Freemarker或Velocity用于模板引擎,以及JSON、XML等数据格式的支持...

    j2ee代码生成器源代码

    这部分代码可能会使用MyBatis或Hibernate等ORM框架,生成相应的Mapper接口和XML配置文件。 4. **Service层生成**:服务层代码处理业务逻辑,调用DAO进行数据操作,并可能包含事务管理。通常会有Service接口和实现类...

    自动生成代码模块注册

    这部分可能涉及到解析配置文件、读取数据库或其他数据源来获取模块信息。 4. **集成**:生成的代码需要无缝地融入到现有项目中。这可能需要工具能够定位到项目的合适位置,如特定的代码文件或目录,以及适应项目的...

Global site tag (gtag.js) - Google Analytics