freeMarker自定义指令

FreeMarker是一个模板引擎,一个基于模板生成文本输出的通用工具,使用纯Java编写。FreeMarker被设计用来生成HTMLWeb页面,特别是基于MVC模式的应用程序。

 所谓模板,就是一份已经写好了基本内容,有着固定格式的文档,其中空出或者用占位符标识的内容,由使用者来填充,不同的使用者给出的数据是不同的。在模板中的占位符,在模板运行时,由模板引擎来解析模板,并采用动态数据替换占位符部分的内容。
   FreeMarker不是一个Web应用框架,而适合作为Web应用框架一个组件,FreeMarker与Web容器无关,即在Web运行时,它并不知道Servlet或HTTP。它不仅可以用作表现层的实现技术,而且还可以用于生成XML,JSP或Java文件等。
 虽然FreeMarker具有一些编程的能力,但通常由Java程序准备要显示的数据,由FreeMarker生成页面,通过模板显示准备的数据

分离表现层和业务逻辑
使用JSP开发过程中在页面中大量的存在业务逻辑的代码,使得页面内容凌乱,在后期大量的修改维护过程中就变得非常困难。FreeMarker根本不支持Java脚本代码,而是使用el表达式来输出展示数据。FreeMarker的设计初衷就是:模板+数据模型=输出,模板只负责数据在页面中的表现,不涉及任何的逻辑代码,而所有的逻辑都是由数据模型来处理的。用户最终看到的输出是模板和数据模型合并后创建的。

提高开发效率
在我们以往的开发中,使用的都是JSP页面来展示数据的,即所谓的表现层。我们都知道,JSP在第一次执行的时候需要转换成Servlet类,开发阶段进行功能调适时,需要频繁的修改JSP,每次修改都要编译和转换,那么试想一天中我们浪费在程序编译的时间有多少。相对于JSP来说,FreeMarker模板技术不存在编译和转换的问题,所以就不会存在上述问题。而且开发过程中,我们在不必在等待界面设计开发人员完成页面原形后,我们再来开发程序。

分工明确
以往用JSP展现数据时,程序员并不熟悉界面设计技术,反之界面开发人员,也并不熟悉程序语言。协调工作很困难,使用FreeMarker后,作为界面开发人员,只专心创建HTML文件、图像以及Web页面的其他可视化方面,不用理会数据;而程序开发人员则专注于系统实现,负责为页面准备要显示的数据。

自定义指令

1、编写一个HelloDirective 实现 TemplateDirectiveModel接口
2、重写execute方法:

    a).解析参数:BeansWrapper wrapper = new BeansWrapperBuilder(Configuration.VERSION_2_3_21).build();   
        Integer obj = (Integer)wrapper.unwrapper(Template model, Integer.class);
    b). 将结果存入env = request.setAbbtribute(key, obj): env.setVariable("hello_world", model);
    c). 返回到页面:
        if (body == null) {
            env.getOut().write(JSON.toJsonString(obj));
        } else {
            body.render(env.getOut());
        }

import com.alibaba.fastjson.JSON;
import freemarker.core.Environment;
import freemarker.ext.beans.BeansWrapperBuilder;
import freemarker.template.*;
import freemarker.ext.beans.BeansWrapper;
import org.springframework.stereotype.Component;

import java.io.IOException;
import java.util.HashMap;
import java.util.Map;

/**
 * Created by JY on 2017/7/13.
 */
@Component
public class HelloDirective implements TemplateDirectiveModel {

    /**
     *
     * @param env   环境
     * @param params    参数
     * @param loopVars  变量
     * @param body  指令输出的内容
     * @throws TemplateException
     * @throws IOException
     */
    @Override
    public void execute(Environment env, Map params, TemplateModel[] loopVars,
                        TemplateDirectiveBody body) throws TemplateException, IOException {

        System.out.println("参数:" + params);

        // 转化器 通过调用wrap方法将java对象转化为freemarker的TemplateModel
        // 调用unwrap方法将TemplateModel转化为Java对象
        BeansWrapper beansWrapper = new BeansWrapperBuilder(Configuration.VERSION_2_3_21).build();
        TemplateModel abc = (TemplateModel) params.get("abc");
        Integer abcInt = (Integer) beansWrapper.unwrap(abc,Integer.class);
        //将参数存入result
        TemplateModel name = (TemplateModel) params.get("name");
        String nameStr = (String) beansWrapper.unwrap(name,String.class);

        Map<String,String> result = new HashMap<>();
        result.put("userName",nameStr);

        TemplateModel model = beansWrapper.wrap(result);
        // 设置值==request.setAttribute(key, value)
        env.setVariable("hello_world", model);

        //输出
        if(body == null){
            env.getOut().write(JSON.toJSONString(result));
        }else{
            body.render(env.getOut());
        }
    }
}

3、在spring中定义模板名称

    <bean id="freemarkerConfig"
      class="org.springframework.web.servlet.view.freemarker.FreeMarkerConfigurer">
    <property name="templateLoaderPath" value="/WEB-INF/views/" />
    <property name="defaultEncoding" value="UTF-8" />
    <property name="freemarkerSettings">
        <props>
            <prop key="number_format">0.##########</prop>
        </props>
    </property>
<!--定义模板名称-->
    <property name="freemarkerVariables">
        <map>
            <entry key="hello_world" value-ref="helloDirective" />
        </map>
    </property>
</bean>

4、页面引用



<@hello_world name=” name> {helloworld.userName} : HELLO WORLD!!

二、全局设置

在application.xml中的视图配置

<bean id="freemarkerConfig"
          class="org.springframework.web.servlet.view.freemarker.FreeMarkerConfigurer">
        <property name="templateLoaderPath" value="/WEB-INF/views/" />
        <property name="defaultEncoding" value="UTF-8" />
        <property name="freemarkerSettings">
            <props>
                <prop key="number_format">0.##########</prop> <!--解决数字问题-->
                <prop key="boolean_format">true,false</prop> <!--解决页面布尔值的输出-->
                <prop key="tag_syntax">auto_detect</prop> <!-- auto_detect:自动选择(选择第一种标签语法) angle_bracket(<#if>) square_bracket[#if]-->
                <prop key="classic_compatible">true</prop> <!--非空不会报错-->
                <prop key="template_update_delay">0</prop> <!--缓存时间-->
                <prop key="default_encoding">UTF-8</prop>
            </props>
        </property>
        <property name="freemarkerVariables">
            <map>
                <entry key="hello_world" value-ref="helloDirective" />
            </map>
        </property>

    </bean>
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值