该帖已经被评为良好帖
|
|
---|---|
作者 | 正文 |
发表时间: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} 至此整个例子就结束了,以上只是最简单的介绍,当然这两种技术还有待我们的深入研究。这个例子只不过是比较直观的表现两种技术的使用而已 而且如果想学习方法模板模式和工厂模式的同学可以下载代码看看 作者:张荣华,未经作者同意不得随意转载! 声明:ITeye文章版权属于作者,受法律保护。没有作者书面许可不得转载。
推荐链接
|
|
返回顶楼 | |
发表时间:2007-04-16
楼主辛苦了。
不过我更喜欢Velocity 不说它已经是APACHE的顶级项目, 我很喜欢它不支持HTML标签这个做法,可以让我更方便的生成JSP页面。呵呵。。。 另外,貌似FREEMAKER是有BUG的。我没用过,知道的朋友请来说一下。 |
|
返回顶楼 | |
发表时间:2007-04-23
sg552 写道 另外,貌似FREEMAKER是有BUG的。我没用过,知道的朋友请来说一下。 大多数框架都是有bug的,关键看这个bug是否对你有影响了,webwork2开始就以freemarker作模板了,现在的struts2.0也是的 |
|
返回顶楼 | |
发表时间:2007-04-23
velocity也是可以format数字 日期嘛 好像要自己写了 没找过 不清楚
至于第2点 呵呵.......... 各有所好吧 作为一个优缺点来说好像有点不适合 一直用emeidt写这东西 颜色嘛 随便改了 不知道ECLIPSE的插件除了变色还有啥功能 有自动完成么? 如果有的话还算是有点吸引力 他的文本编辑功能太弱了 velocity最讨厌的一点是代码写错的时候debug比较困难 freemarker没用过 不评价 |
|
返回顶楼 | |
发表时间:2007-04-23
另外 velocity的调用不用你那么麻烦的 你这个例子明显复杂了很多 一般只有在不返回页面 用到他的模板功能的时候才这么用
如果当成一个页面来写 完全不至于写成这样 |
|
返回顶楼 | |
发表时间:2007-04-23
ddandyy 写道 另外 velocity的调用不用你那么麻烦的 你这个例子明显复杂了很多 一般只有在不返回页面 用到他的模板功能的时候才这么用
如果当成一个页面来写 完全不至于写成这样 你说的没错,用在view层的时候,很多web框架都有很好的支持,但是我想讨论得更多,因为模板技术的价值不只在view层,很多地方都可以用,比如在新闻系统中生成静态页面等等,还有我上面提到的等等 |
|
返回顶楼 | |
发表时间:2007-04-23
新闻系统中生成静态页面,我怎么看也是在view层的应用呢
模板的编辑一般都比较简单,只提供数据的合成,最难在调试上 模板在iew层上不能直接看到真是的效果,要是有填充数据,或支持直接预览就方便多了 |
|
返回顶楼 | |
发表时间:2007-04-23
wensky222 写道 新闻系统中生成静态页面,我怎么看也是在view层的应用呢
不会啊,新闻系统中生成静态页面后就不管模板引擎的事情了,接下来就靠servlet容器或apache直接解析html页面了,模板引擎只管生成静态页面,所以这个不算view层的应用哦 |
|
返回顶楼 | |
发表时间: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菜鸟刚飞,一家之言,请各位前辈多多批评! |
|
返回顶楼 | |
发表时间:2007-05-18
velocity没有macro,这点比freemarker差多了,而且才是二者差别最显著的地方。
|
|
返回顶楼 | |