velocity模板渲染的步骤:

1) 首先初始化启动Velocity引擎,可以通过Velocity.init()或者新建VelocityEngine类,并调用其中的init()方法;

2) 创建一个VelocityContext对象,将变量名与值关联起来,与HashMap对象相类似。可以直接将值传递给页面进行引用;

3) 获取一个模板,利用Velocity.getTemplate()获取一个渲染模板,即要将数据最终渲染在哪个页面上去。

4) 创建一个输出流,将上述创建的数据最终渲染到模板上,采用的方法template.merge()

代码示例:

try {
Velocity.init("velocity.properties");
VelocityContext context = new VelocityContext();
String templateFile = "template.vm";
context.put("paramObject", "onlyone");
Template template = null;
template = Velocity.getTemplate(templateFile);
BufferedWriter writer = new BufferedWriter(new FileWriter("velocity.data"));
if (template != null) template.merge(context, writer);
writer.flush();
writer.close();
} catch (Exception e) {
e.printStackTrace();
}

原理分析:


整体工作流程图:

1. Velocity引擎初始化及启动

Velocity引擎在启动时,无论是采用Velocity.init()还是采用新建VelocityEngine的方式均会调用RuntimeInstance类中的init方法,通过设置的引擎属性初始化引擎,包括国际化支持,ResourceLoader设置,字符编码来完成一些属性资源及原始指令的加载工作。

1)initializeProperties()方法:初始化velocity的全局属性信息,即velocity.properties中配置的信息。首先会加载velocity自身所带的一个全局velocity.properties文件(org\apache\velocity\runtime\defaults\velocity.properties),若用户自己有写velocity.properties文件,则会将其中已经加载的一些属性覆盖掉。

2)initializeLog()方法:初始化velocity的日志信息。

3)initializeResourceManager()方法:初始化velocity的资源管理。

4)initializeDirectives()方法:初始化velocity中的语法法则。

5)initializeEventHandlers()方法:初始化事件句柄。

6)initializeParserPool()方法:初始化velocity的解析工具。

7)initializeIntrospection()方法:根据所获取到的uberspect名称实例化类。

8)vmFactory.initVelocimacro()方法:实例化VM工厂,它将从运行时中获取属性值,并初始化所有的宏。

2. 查找并解析模板

通过资源加载器(ResourceLoader)将模板文件加载到内存(转化为InputStream),然后通过AST(Abstract Syntax Tree)解析器将InputStream解析为一个AST。上面的代码,当执行Velocity.getTemplate("template.vm")时,首先通过ResourceLoader将以字节流的形式加载template.vm文件,然后通过Parser生成如下Token集合:{[<html> <body> Hello], [$foo], [world! </body> </html> ]},可以发现velocity根本不关心模板最终要渲染出来的是html还是其他什么东西,在此处对于velocity而言html标签意味着就是纯文本标签。最终构建的AST如下:

根节点下有三个子节点:

[<html> <body> Hello]    对应ASTText节点(纯文本节点)

[$foo]  对应ASTReference节点(需要替换的引用节点)

[world! </body> </html> ] 对应ASTText节点 (纯文本节点)

Velocity引擎在这里有个优化策略,可以针对生成的语法树进行cache

 

深入分析:


  • Velocity为其模板语法定义了一份jjt文件,根据这份jjt文件,使用JJTree生成一个语法解析器。
  • Velocity将模板解析的过程完全交给了语法解析器,调用解析器的parse方法直接得到AST,这一棵AST的每一个节点都对应一个SimpleNode的子类,其中不同的语法元素对应的不同的SimpleNode,比如#if条件表达式对应的SimpleNode是ASTIfStatement,而#stop指令对应的SimpleNode是ASTStop。
  • 得到了AST以后,模板的渲染就比较简单了,无非就是递归地调用各个节点的SimpleNode的render方法来完成模板的渲染过程。

在了解了模板的渲染过程后,那模板上的方法是如何在渲染过程中执行的?对于Velocity里面的引用,比如$Person这样的,最后都被解析成AST中的ASTReference节点,而对于$Person.name这样的,ASTReference下面有一个ASTIdentifier节点,$Person.saySomething()这样的,ASTReference下面有一个ASTMethod节点。不论是ASTMethod还是ASTIdentifier,最后都是通过Uberspect和Introspector这两个类来完成对方法的查找,最后调用各种Executor来实现对方法的调用。

Uberspect这个类的功能是通过反射(Reflection)和内省(Introspection)来完成对需要调用的方法的获取的,而Introspector这个类的功能是根据方法名和方法参数在一个类中查找Method对象。另外,为了提高性能,会对Method的数据进行了缓存(见IntrospectionCacheData,IntrospectorCache和IntrospectorCacheImpl三个类),以便下次快速可以找到。

找到Method以后,具体的方法的执行由各个Executor控制,每一个Executor都继承了AbstractExecutor,给外部提供统一的接口去调用。

3.上下文VelocityContext

将k--v属性对注入到上下文参数中

 VelocityContext context = new VelocityContext();
context.put("paramObject", "onlyone");

4. 模板渲染,输出字符流

当执行template.merge(context, writer);时,模板遍历其对应的AST树,执行每个节点的渲染过程。如ASTText节点只是简单的将文本写入writer。ASTReference节点需要从context中获取引用的参数paramObject的值onlyone将${paramObject}替换,并写入到writer中。Velocity的AST中有多种节点,如ASTIdentitor等,有些需要反射机制处理。当整个AST遍历结束,也就意味着模板渲染结束,渲染的结果写入writer流中。
Spring与Velocity的整合


Spring MVC是基于DispatcherServlet拦截请求并找到相应的控制器进行业务逻辑处理。Velocity与Spring整合主要需要配置的一些文件。
web.xml
    <!--Spring 服务层的配置文件 -->
<context-param>
<param-name>contextConfigLocation</param-name>
<param-value>classpath:applicationContext.xml</param-value>
</context-param> <!--Spring 容器启动监听器 -->
<listener>
<listener-class>org.springframework.web.context.ContextLoaderListener</listener-class>
</listener> <!-- Spring MVC 的Servlet,它将加载WEB-INF/springDispatcher-servlet.xml 的配置文件,以启动Spring MVC模块-->
<servlet>
<servlet-name>springDispatcher</servlet-name>
<servlet-class>
org.springframework.web.servlet.DispatcherServlet
</servlet-class>
<init-param>
<param-name>contextConfigLocation</param-name>
<param-value>
/WEB-INF/springDispatcher-servlet.xml
</param-value>
</init-param>
</servlet> <servlet-mapping>
<servlet-name>springDispatcher</servlet-name>
<url-pattern>*.do</url-pattern>
</servlet-mapping>

springDispatcher-servlet.xml中的配置信息:

<--1、对模型视图名称的解析,即在模型视图名称添加前后缀 -->
<bean id="viewResolver" class="org.springframework.web.servlet.view.velocity.VelocityViewResolver">
<property name="allowSessionOverride" value="true"/>
<property name="exposeSessionAttributes" value="true"/>
<property name="cache" value="true"/>
<property name=”prefix” value=”/WEB-INF/templates/”/>
<property name="suffix" value=".vm"/>
<property name="contentType">
<value>text/html;charset=UTF-8</value>
</property>
</bean> <--2、velocity的一些设置 -->
<bean id="velocityConfig" class="org.springframework.web.servlet.view.velocity.VelocityConfigurer">
<property name="resourceLoaderPath">
<value>velocity/</value>
</property>
<property name="velocityProperties">
<props>
<prop key="input.encoding">UTF-8</prop>
<prop key="output.encoding">UTF-8</prop>
<prop key="contentType">text/html;charset=UTF-8</prop>
</props>
</property>
</bean>

1处的工作即是定义模型视图名称的解析规则,这里我们采用的是velocity模板视图解析器,这样Spring就与velocity模板技术整合起来了。Spring的Dispatcher中有一个属性为viewResolver,通过它来寻找真正的视图对象。
2处定义了velocity的一些属性配置,包括资源的加载路径和页面的编码格式等。

参考资料:

Velocity内部实现的介绍:http://agapple.iteye.com/blog/1071438
velocity引擎源码学习一:http://bojiang.iteye.com/blog/1159313
velocity中的Introspection介绍:http://bojiang.iteye.com/blog/1159293
velocity的优化记录:http://agapple.iteye.com/blog/1051724

Velocity源码分析的更多相关文章

  1. [Android实例] Scroll原理-附ScrollView源码分析

    想象一下你拿着放大镜贴很近的看一副巨大的清明上河图, 那放大镜里可以看到的内容是很有限的, 而随着放大镜的上下左右移动,就可以看到不同的内容了 android中手机屏幕就相当于这个放大镜, 而看到的内 ...

  2. BIZ中model.getSql源码分析

    功能:根据model.xml文件中配置的sql,获取对应的动态sql结果. 实例代码:String sql1 = model.getSql(dao.dbMeta());String sql2 = mo ...

  3. springMVC源码分析--视图View(一)

    之前的博客springMVC源码分析--HttpMessageConverter数据转化(一)中我们已经介绍了数据返回值的处理,在博客springMVC源码分析--ViewResolver视图解析器( ...

  4. Facebook Rebound 弹性动画库 源码分析

    Rebound源码分析 让动画不再僵硬:Facebook Rebound Android动画库介绍一文中介绍了rebound这个库. 对于想体验一下rebound的效果,又懒得clone和编译代码的, ...

  5. [Android实例] Scroll原理-附ScrollView源码分析 (转载)

    想象一下你拿着放大镜贴很近的看一副巨大的清明上河图, 那放大镜里可以看到的内容是很有限的, 而随着放大镜的上下左右移动,就可以看到不同的内容了 android中手机屏幕就相当于这个放大镜, 而看到的内 ...

  6. 精尽Spring MVC源码分析 - 寻找遗失的 web.xml

    该系列文档是本人在学习 Spring MVC 的源码过程中总结下来的,可能对读者不太友好,请结合我的源码注释 Spring MVC 源码分析 GitHub 地址 进行阅读 Spring 版本:5.2. ...

  7. ABP源码分析一:整体项目结构及目录

    ABP是一套非常优秀的web应用程序架构,适合用来搭建集中式架构的web应用程序. 整个Abp的Infrastructure是以Abp这个package为核心模块(core)+15个模块(module ...

  8. HashMap与TreeMap源码分析

    1. 引言     在红黑树--算法导论(15)中学习了红黑树的原理.本来打算自己来试着实现一下,然而在看了JDK(1.8.0)TreeMap的源码后恍然发现原来它就是利用红黑树实现的(很惭愧学了Ja ...

  9. nginx源码分析之网络初始化

    nginx作为一个高性能的HTTP服务器,网络的处理是其核心,了解网络的初始化有助于加深对nginx网络处理的了解,本文主要通过nginx的源代码来分析其网络初始化. 从配置文件中读取初始化信息 与网 ...

随机推荐

  1. JavaScript自学代码--(四)

    //JavaScript Window - 浏览器对象模型 window.document.getElementById("header"); //等价于 document.get ...

  2. uva 1449 - Dominating Patterns

    简单的AC自动机: #include<cstdio> #include<cstring> #include<queue> #define maxn 150005 u ...

  3. 【Tools】Chrome开发者工具详解

    作为一名前端开发者,打交道最多的可能是和浏览器.市面上各种浏览器多不胜数,主流的有Chrome,Firefox,Safari,IE,Opera,非主流的如360,遨游,QQ浏览器,搜狗浏览器,据说淘宝 ...

  4. Android用户界面 UI组件--AdapterView及其子类(二) AdapterViewAnimator及其子类

    AdapterViewAnimator:当在视图间切换时会显示动画. android:animateFirstView 定义ViewAnimation首次显示时是否对当前视图应用动画. android ...

  5. Android 获取SDCard中某个目录下图片

    本文介绍Android开发中如何获取SDCard中某目录下的所有图片并显示出来,下面的我们提供的这个函数是通用的,只要提供路径就可以查询出该目录下所有图片的路径信息,并保存到一个List<Str ...

  6. Serif和Sans-serif字体的区别(转)

    在西方国家罗马字母阵营中,字体分为两大种类:Sans Serif和Serif,打字机体虽然也属于Sans Serif,但由于是等宽字体,所以另外独立出Monospace这一种类,例如在Web中,表示代 ...

  7. eMMC(KLM8G2FE3B)

     Tiny4412原理图中,eMMC是169-PIN,资料中对应内存为16/32G:而用户手册上eMMC内存为4G,对应的是153-PIN?    原理图中上标注:KLM8G2FE3B-B001_1. ...

  8. UVA_393_Doors_(计算几何基础+最短路)

    描述 https://uva.onlinejudge.org/index.php?option=com_onlinejudge&Itemid=8&category=5&page ...

  9. C++运行字符编码于MSVC和GCC之间的区别

    详细请参考这篇博文 http://blog.csdn.net/dbzhang800/article/details/7540905 运行字符编码就是指,当你源代码写下const char* p = & ...

  10. win7 下安装oracle 10g

    oracle 10g 在win7下安装,提示程序异常终止,发生未知错误 在网上搜结果: 修改Oracle 10G\database\stage\prereq\db\refhost.xml 在 < ...