论坛首页 Java企业应用论坛

模板:velocity和freemarker的比较

浏览 59032 次
该帖已经被评为良好帖
作者 正文
   发表时间:2007-04-16  
/**
*作者:张荣华(ahuaxuan)
*2007-04-16
*转载请注明出处及作者
*/

模板技术在现代的软件开发中有着重要的地位,而目前最流行的两种模板技术恐怕要算freemarker和velocity了,webwork2.2对两者都有不错的支持,也就是说在webwork2中你可以随意选择使用freemarker或velocity作为view,模板技术作为view的好处是很多,尤其和jsp比较起来优点更大,众所周知jsp需要在第一次被执行的时候编译成servlet,那么这个过程是很慢的,当然很多应用服务器都提供预编译的功能,但是在开发的时候仍然给我们程序员带来了很多痛苦,每次修改都要多几秒钟,那在一天的开发中就有很多时间浪费在jsp的编译上了。用webwork in action的作者的话来说:“每次修改之后重新运行都要等等几秒是令人失望的,而频繁地修改jsp更是会令你的失望情绪变本加厉“。我们把模板技术引入到view中去可以带来更好的开发效率,而且模板的速度要比jsp快(虽然编译过后的jsp在速度上已经满足我的需求了,呵呵)。 当然模板技术可以用在很多领域,可不只在view那里。我们可以通过模板技术来生成xml,生成jsp,生成java文件等等,说到这里,大家通常会使用模板技术用在公司的框架里,这样就可以很快速的生成添删改查的代码,需要的只是模板,其他比如还有邮件模板等等。

以上是模板的作用,当然模板还有其他领域的应用,希望能和大家多讨论,提高我们的生产效率。

那么现在开源的模板技术有好几种,多了之后就有一个选择的问题了,如何选择一个满足自己需要的模板的呢,我大概了看了一下两种模板技术,写了一个例子,我使用了几种设计模式来完成了这个例子,这个例子中,我同时使用了freemarker和velocity,这样同学们可以通过代码很直观的比较两种模板技术,通过这个例子,我认识到freemarker在功能上要比velocity强大
1在view层的时候,它提供了format日期和数字的功能,我想大家都有在页面上format日期或数字的经验,用jsp的同学可能对jstl的fmt标签很有感情,使用了freemarker之后也可以使用freemarker提供的功能来formmat日期和数据,这个功能我想是很贴心的

2通过我的使用我发现freemaker的eclipseplugin要比velocity的eclipseplugin好,如果你是用idea那很遗憾,我没有找到类似的插件。好在很多地方呢,我看到的是freemarker的插件除了支持freemarker语法也支持html语句,而velocity的插件貌似只支持velocity的语法,html就只是用普通的文本来显示了,在这一点上freemarker占上风了(不要和我说高手都是用windows记事本之类的话,这本来就违背了模板技术的初衷)

3freemarker对jsptag的支持很好,算了,不到迫不得已还是不要这样做吧。

还有就是两者的语法格式,这一点上不同的人有不同倾向

下面就介绍一下这个例子吧

/**
 * 
 * @author 张荣华
 * 转载请注明出处
 */
public class TemplateTest {

	/**
	 * @param args
	 */
	public static void main(String[] args) throws Exception{
		/* 准备数据 */
        Map latest = new HashMap();
        latest.put("url", "products/greenmouse.html");
        latest.put("name", "green mouse");
        
        Map root = new HashMap();
        root.put("user", "Big Joe");
        root.put("latestProduct", latest);
        root.put("number", new Long(2222));
        root.put("date",new Date());
        
        List listTest = new ArrayList();
        listTest.add("1");
        listTest.add("2");
        
        root.put("list",listTest);
        
        TemplateEngine freemarkerEngine = (TemplateEngine)TemplateFactory.getInstance().getBean("freemarker");
        freemarkerEngine.run(root);//使用freemarker模板技术
        
        TemplateEngine velocityEngine = (TemplateEngine)TemplateFactory.getInstance().getBean("velocity");
        velocityEngine.run(root);//使用velocity模板技术
	}

}


工厂类,用来得到模板引擎
/**
 * 
 * @author 张荣华
 * 转载请注明出处
 */
public class TemplateFactory {
	private static TemplateFactory instance;
	private Map objectMap;
	
	static{
		instance = new TemplateFactory();
	}
	
	public TemplateFactory() {
		super();
		this.objectMap = new HashMap();
		synchronized (this) {
			objectMap.put("freemarker", new FreemarkerTemplateEngine(){
				public String getTemplatePath() {
					return "template";
				}
			});
			
			objectMap.put("velocity", new VelocityTemplateEngine());
		}
	}

	public static TemplateFactory getInstance(){
		return instance;
	}
	
	/**
	 * 模仿spring的工厂
	 * @param beanName
	 * @return
	 */
	public Object getBean(String beanName){
		return objectMap.get(beanName);
	}

}
引擎接口
/**
 * 
 * @author 张荣华
 * 转载请注明出处
 */
public interface TemplateEngine {
	
	void run(Map context)throws Exception;

}
模板引擎的实现使用method template模式,因为有两个实现,这两个实现又存在公共的逻辑,所以选择了这个模式

/**
 * 
 * @author 张荣华
 * 转载请注明出处
 */
public abstract class AbstractTemplateEngine implements TemplateEngine{

	public abstract String getTemplatePath();
	
	public abstract String getTemplate();
	
	public abstract String getEngineType();
	
	public void run(Map context)throws Exception{
		if(Constants.ENGINE_TYPE_FREEMARKER.equals(getEngineType()))
			executeFreemarker(context);
		else
			executeVelocity(context);
	}
	
	private void executeFreemarker(Map context)throws Exception{
		Configuration cfg = new Configuration();
	    cfg.setDirectoryForTemplateLoading(
	            new File(getTemplatePath()));
	    cfg.setObjectWrapper(new DefaultObjectWrapper());
	    
	    cfg.setCacheStorage(new freemarker.cache.MruCacheStorage(20, 250));
	            
	    Template temp = cfg.getTemplate(getTemplate());

	    Writer out = new OutputStreamWriter(System.out);
	    temp.process(context, out);
	    out.flush();
	}
	
	private void executeVelocity(Map root)throws Exception{
		
		Velocity.init();
		VelocityContext context = new VelocityContext(root);
		org.apache.velocity.Template template = null;
		
	    template = Velocity.getTemplate(getTemplatePath()+getTemplate());
	    
		StringWriter sw = new StringWriter();
		template.merge( context, sw );
		System.out.print(sw.toString());

	}

}



这个是freemarker实现
/**
 * 
 * @author 张荣华
 * 转载请注明出处
 */
public class FreemarkerTemplateEngine extends AbstractTemplateEngine{
	private static final String DEFAULT_TEMPLATE = "FreemarkerExample.ftl";
	
	/**
	 * 这个方法应该实现的是读取配置文件
	 */
	public String getTemplatePath() {
		return null;
	}
	
	public void run(Map root) throws Exception{
		super.run(root);
	}

	public String getTemplate() {
		// TODO Auto-generated method stub
		return DEFAULT_TEMPLATE;
	}

	public String getEngineType() {
		return Constants.ENGINE_TYPE_FREEMARKER;
	}
}
这个是velocity实现
/**
 * 
 * @author 张荣华
 * 转载请注明出处
 */
public class VelocityTemplateEngine extends AbstractTemplateEngine{

private static final String DEFAULT_TEMPLATE = "VelocityExample.vm";

	public String getTemplatePath() {
		return "/template/";
	}
	
	public void run(Map map) throws Exception{
		super.run(map);
	}

	public String getTemplate() {
		// TODO Auto-generated method stub
		return DEFAULT_TEMPLATE;
	}

	public String getEngineType() {
		// TODO Auto-generated method stub
		return Constants.ENGINE_TYPE_VELOCITY;
	}
}


以下是模板
1,freemarker模板
freemarker template test:
string test-----------${user}-----------${number}-----------${latestProduct.url}-----------${latestProduct.name}
condition test-----------
<#if user == "Big Joe">
list iterator-----------
<#list list as aa>
${aa}
</#list> 
</#if>
date test-----------${date?string("MMM/dd/yyyy")}


2,velocity模板

******************************************************************************************************************
velocity template test:
string test-----------${user}-----------${number}-----------${latestProduct.url}-----------${latestProduct.name}
condition test-----------
#if ($user == "Big Joe")
list iterator-----------
#foreach( $aa in $list )
$aa
#end
#end
date test-----------${date}


至此整个例子就结束了,以上只是最简单的介绍,当然这两种技术还有待我们的深入研究。这个例子只不过是比较直观的表现两种技术的使用而已
而且如果想学习方法模板模式和工厂模式的同学可以下载代码看看

作者:张荣华,未经作者同意不得随意转载!
  • template.rar (2.1 MB)
  • 描述: 可运行代码
  • 下载次数: 3275
   发表时间:2007-04-16  
楼主辛苦了。

不过我更喜欢Velocity

不说它已经是APACHE的顶级项目,

我很喜欢它不支持HTML标签这个做法,可以让我更方便的生成JSP页面。呵呵。。。

另外,貌似FREEMAKER是有BUG的。我没用过,知道的朋友请来说一下。
0 请登录后投票
   发表时间:2007-04-23  
sg552 写道

另外,貌似FREEMAKER是有BUG的。我没用过,知道的朋友请来说一下。

大多数框架都是有bug的,关键看这个bug是否对你有影响了,webwork2开始就以freemarker作模板了,现在的struts2.0也是的
0 请登录后投票
   发表时间:2007-04-23  
velocity也是可以format数字 日期嘛 好像要自己写了 没找过 不清楚
至于第2点 呵呵.......... 各有所好吧  作为一个优缺点来说好像有点不适合
一直用emeidt写这东西 颜色嘛 随便改了
不知道ECLIPSE的插件除了变色还有啥功能  有自动完成么? 如果有的话还算是有点吸引力 他的文本编辑功能太弱了

velocity最讨厌的一点是代码写错的时候debug比较困难
freemarker没用过 不评价
0 请登录后投票
   发表时间:2007-04-23  
另外 velocity的调用不用你那么麻烦的 你这个例子明显复杂了很多  一般只有在不返回页面 用到他的模板功能的时候才这么用
如果当成一个页面来写  完全不至于写成这样
0 请登录后投票
   发表时间:2007-04-23  
ddandyy 写道
另外 velocity的调用不用你那么麻烦的 你这个例子明显复杂了很多  一般只有在不返回页面 用到他的模板功能的时候才这么用
如果当成一个页面来写  完全不至于写成这样

你说的没错,用在view层的时候,很多web框架都有很好的支持,但是我想讨论得更多,因为模板技术的价值不只在view层,很多地方都可以用,比如在新闻系统中生成静态页面等等,还有我上面提到的等等
0 请登录后投票
   发表时间:2007-04-23  
新闻系统中生成静态页面,我怎么看也是在view层的应用呢

模板的编辑一般都比较简单,只提供数据的合成,最难在调试上

模板在iew层上不能直接看到真是的效果,要是有填充数据,或支持直接预览就方便多了
0 请登录后投票
   发表时间:2007-04-23  
wensky222 写道
新闻系统中生成静态页面,我怎么看也是在view层的应用呢

不会啊,新闻系统中生成静态页面后就不管模板引擎的事情了,接下来就靠servlet容器或apache直接解析html页面了,模板引擎只管生成静态页面,所以这个不算view层的应用哦
0 请登录后投票
   发表时间:2007-05-18  
我觉得各个技术都有自己的最适合位置。

在根据模板生成代码方面,velocity,  freemaker确实很强。 从apache的顶级子项目来看, velocity要比freemaker有前途。

在事件响应的方面,JSF, Tapestry 也确实做的比较好。JSF是SUN的官方首推,Tapestry也是apache的顶级子项目,它们都有支持。

但是如果要说 Velocity , Freemaker 将取代JSP的地位,我不敢苟同。我认为就连JSF, tapestry 也无法动摇JSP的地位。 上面四种技术,虽然我不精通,但都有使用经验。我的感觉是:

velocity, freemaker, 用来生成工作代码,尤其是JSP代码,很方便。另外使用的时候,需要遵循“非常严格”的MVC。JSF,Tapestry适合那些习惯于C/S或者桌面应用开发的程序员,完全可以不懂HTML,就能使用组件。我的感觉,是对JSP的一个大补丁。



话题回来,我的意见是:想做好B/S系统,JSP是绝对的武功正宗,潜心修炼是没错的。v, m ,jsf, t 和林林种种其他功夫,作为辅助效果不错,但优势只是一时,越是用到后来,越是难以修炼。

ss菜鸟刚飞,一家之言,请各位前辈多多批评!
1 请登录后投票
   发表时间:2007-05-18  
velocity没有macro,这点比freemarker差多了,而且才是二者差别最显著的地方。
0 请登录后投票
论坛首页 Java企业应用版

跳转论坛:
Global site tag (gtag.js) - Google Analytics