《FreeMarker教程-笔录》

今天我们学习一下FreeMarker模板引擎。它是基于模板文件生成其他文本的通用工具。本章我们主要讲使用FreeMarker模板引擎生成 .html 文件和生成 .java 类文件。

简介

FreeMarker是一款用java语言编写的模版引擎,它虽然不是web应用框架,但它很适合作为web应用框架的一个组件。
特点:

  • 轻量级模版引擎,不需要Servlet环境就可以很轻松的嵌入到应用程序中
  • 能生成各种文本,如html,xml,java,等
  • 入门简单,它是用java编写的,很多语法和java相似

环境配置

首先我们需要配置FreeMarker环境,两种方式任选其一(推荐maven):
1.下载它的依赖jar包
2.用maven配置。

  • jar包下载地址:http://vdisk.weibo.com/s/teaEtjrsmozAQ
  • maven方式配置:

    org.freemarker
    freemarker
    2.3.20

--------------好了,环境配好了我们开始搞点事情吧!

通过FreeMarker的模板文件生成java类文件

1.新建一个javaweb工程(eclipse/idea都可以)
2.在资源目录下新建一个templates目录(用来存放我们所有的FreeMarker模板文件xxx.ftl,模板文件的后缀名称为 ".ftl")
3.新建一个模板文件 hello.ftl (文件内容):

package ${classPath};//模板中变量引用(动态赋值)

public class ${className} {

    public static void main(String[] args) {
        System.out.println("${helloWorld}");
    }

}

4.在java类的src目录包下(包名随便【不过需要注意的是要和代码里生成的包位置一致】)新建FreemarkerTest.java (文件内容):

package com.yu.scloud.baseframe.frame.freemarker;

import com.yu.scloud.baseframe.frame.pojo.User;
import freemarker.template.Configuration;
import freemarker.template.Template;

import java.io.*;
import java.util.*;

public class FreemarkerTest {

    private static final String TEMPLATE_PATH = "src/main/resources/templates";//模板路径【指向你自己的模板文件存放路径】

    public static void main(String[] args) {

        autoGenJAVATest();//通过.ftl模板生成java类

    }
    //根据模板自动创建java类-示例
    private static void autoGenJAVATest()
    {
        Map dataMap = new HashMap();
        dataMap.put("classPath", "com.yu.scloud.baseframe.frame.freemarker");
        dataMap.put("className", "AutoGenHello");
        dataMap.put("helloWorld", "通过<代码自动生成程序> 演示 FreeMarker的HelloWorld!");

        String DEST_PATH = "src/main/java/com/yu/scloud/baseframe/frame/freemarker";//目标路径【指向你自己的类文件存放路径】
        genFileWithTemplate(TEMPLATE_PATH,DEST_PATH,"hello.ftl","AutoGenHello.java",dataMap);
    }

    //通过.flt模板生成file(封装了一下 方便统一调用)
    private static void genFileWithTemplate(String templateDir,String destDir,String templateFileName,String destFileName,Map dataMap)
    {
        //创建freeMarker配置实例
        Configuration configuration = new Configuration();
        Writer out = null;
        try {
            //设置模版路径
            configuration.setDirectoryForTemplateLoading(new File(templateDir));
            //加载模版文件
            Template template = configuration.getTemplate(templateFileName);
            //生成数据
            File docFile = new File(destDir + "\\" + destFileName);
            out = new BufferedWriter(new OutputStreamWriter(new FileOutputStream(docFile)));
            //输出文件
            template.process(dataMap, out);
            System.out.println(destFileName+" 模板文件创建成功 !");
        } catch (Exception e) {
            e.printStackTrace();
        } finally {
            try {
                if (null != out) {
                    out.flush();
                    out.close();
                }
            } catch (Exception e2) {
                e2.printStackTrace();
            }
        }
    }
}

5.执行 FreemarkerTest.java的main方法,成功后会在上面代码中指定的DEST_PATH指向的目录下生成一个 AutoGenHello.java 文件,(文件内容):

package com.yu.scloud.baseframe.frame.freemarker;

public class AutoGenHello {

    public static void main(String[] args) {
        System.out.println("通过<代码自动生成程序> 演示 FreeMarker的HelloWorld!");
    }

}

以上就是使用FreeMarker模板引擎生成.java类文件的全部过程。我们可以对生成的文件简单分析一下:

  • hello.ftl模板文件中 package ${classPath}; 被替换成了
    package com.yu.scloud.baseframe.frame.freemarker;
  • public class ${className} { 被替换成了
    public class AutoGenHello {
    ......
我们通过模板可以生成多种多样更加复杂的类文件,对于我们日常开发很有帮助。比如Spring框架中的 Controller、Dao、Service以及ServiceImpl,这些类文件我们反复会去创建,并且好多文件格式、方法都是差不多的,这样我们就可以通过模板引擎去一键创建,方便的不要不要滴~~~

接下来我们将使用FreeMarker模板生成html文件

通过FreeMarker的模板文件生成Html文件

其实很简单

  • 新建 hello-html.ftl 文件(用来生成html的模板文件---示例代码中包含了FreeMarker基本的语法使用,东西相对比较多仔细阅读)
  • 我们对上面生成java的那个类改造一下:替换一下你生成html文件的路径以及生成的文件名称:

FreemarkerTest.java类(文件内容):

package com.yu.scloud.baseframe.frame.freemarker;

import com.yu.scloud.baseframe.frame.pojo.User;
import freemarker.template.Configuration;
import freemarker.template.Template;

import java.io.*;
import java.util.*;

public class FreemarkerTest {

    private static final String TEMPLATE_PATH = "src/main/resources/templates";//模板路径

    //通过.ftl模板生成java类
    public static void main(String[] args) {
        
        autoGenHTMLTest();
//        autoGenJAVATest();

    }
    //根据模板自动创建html-示例
    private static void autoGenHTMLTest()
    {
        //给模板文件组装数据
        Map dataMap = new HashMap();
        User user=new User();
        user.setName("Template-FreeMarker");
        user.setId("123456789");
        dataMap.put("user",user);
        dataMap.put("title","FreeMarker自动生成HTML");
        dataMap.put("datetime",new Date());
        //设置列表数据
        List peoples=new ArrayList<>();
        for(int i=0;i<3;i++)
        {
            User user2=new User();
            user2.setName("A"+i);
            user2.setId("a"+i);
            peoples.add(user2);
        }
        dataMap.put("peoples",peoples);
        //设置map数据
        HashMap employees=new HashMap<>();
        for(int m=0;m<6;m++)
        {
            User user3=new User();
            user3.setName("雇员"+m);
            user3.setId("emp"+m);
            employees.put(user3.getId(),user3);
        }
        dataMap.put("employees",employees);

        String DEST_PATH = "src/main/resources/static/demo";//目标路径
        genFileWithTemplate(TEMPLATE_PATH,DEST_PATH,"hello-html.ftl","hello.html",dataMap);
    }

    //通过.flt模板生成file
    private static void genFileWithTemplate(String templateDir,String destDir,String templateFileName,String destFileName,Map dataMap)
    {
        //创建freeMarker配置实例
        Configuration configuration = new Configuration();
        Writer out = null;
        try {
            //设置模版路径
            configuration.setDirectoryForTemplateLoading(new File(templateDir));
            //加载模版文件
            Template template = configuration.getTemplate(templateFileName);
            //生成数据
            File docFile = new File(destDir + "\\" + destFileName);
            out = new BufferedWriter(new OutputStreamWriter(new FileOutputStream(docFile)));
            //输出文件
            template.process(dataMap, out);
            System.out.println(destFileName+" 模板文件创建成功 !");
        } catch (Exception e) {
            e.printStackTrace();
        } finally {
            try {
                if (null != out) {
                    out.flush();
                    out.close();
                }
            } catch (Exception e2) {
                e2.printStackTrace();
            }
        }
    }
}
hello-html.ftl 文件内容:


    
    



    

${title}

名称:${user.name}

年龄:${user.id}

《FreeMarker语法汇总》
声明变量cdate="自定义变量"
<#assign cdate="自定义变量"/>
条件判断
<#if cdate??>
${cdate}
<#else>
变量未声明
字符串截取(原始字符串:"字符串截取Demo"
<#assign str="字符串截取Demo"/>
截取索引为2的字符:${str[2]}
截取区间2-5字符串:${str[2..5]}

算数运算(声明了两个变量a=5,b=8)
<#assign a=5/> <#assign b=8/>
"+":${a+b}
"-":${a-b}
"*":${a*b}
"/":${a/b}
"%":${a%b}

比较运算符
<#if a gte b>a大于等于b
<#if a lt b>a小于b

内建函数
<#assign HELLO="helloMari"/>
第一个字母大写:${HELLO?cap_first}
所有字母小写:${HELLO?lower_case}
所有字母大写:${HELLO?upper_case}
<#assign fnum=3.1415926/>
数值取整(3.1415926):${fnum?int}
日期格式化:${datetime?string("yyyy-MM-dd")}
获取列表长度:${peoples?size}

遍历列表项:
    <#list peoples as people>
  • 人:${people.name}
遍历Map项key:
    <#list employees?keys as key>
  • 键:${key}--值:${employees[key].name}
遍历Map项value:
    <#list employees?values as value>
  • 值:${value.name}
<#assign listData=["ITDragon", "blog", "is", "cool"]>
引入其它ftl模板文件:
<#include "hello-other-html.ftl"/>
宏命令(自定义不带参数的标签)
<#macro customtag>
不带参数宏:${title}
<@customtag/>
宏命令(自定义带参数的标签)
<#macro customtagp p0 p1 p2>
带参数宏:${title}--${p0}--${p1}--${p2}
<@customtagp p0="参数0" p1="参数1" p2="参数2"/> <#import "hello-other-html.ftl" as other/>
${other.otherftl}
hello-other-html.ftl 文件内容:

我是被include的模板内容
<#assign otherftl="另一个ftl文件定义的变量"/>
User.java类内容:
package com.yu.scloud.baseframe.frame.pojo;

import java.io.Serializable;

public class User implements Serializable{

    String id;
    String name;


    public String getId() {
        return id;
    }

    public void setId(String id) {
        this.id = id;
    }

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }
}
执行 FreemarkerTest.java中的main方法,会在目录下生成 hello.html 文件内容(对比一下模板文件):


    
    



    

FreeMarker自动生成HTML

名称:Template-FreeMarker

年龄:123456789

《FreeMarker语法汇总》
声明变量cdate="自定义变量"
条件判断
自定义变量
字符串截取(原始字符串:"字符串截取Demo"
截取索引为2的字符:串
截取区间2-5字符串:串截取D

算数运算(声明了两个变量a=5,b=8)
"+":13
"-":-3
"*":40
"/":0.625
"%":5

比较运算符
a小于b

内建函数
第一个字母大写:HelloMari
所有字母小写:hellomari
所有字母大写:HELLOMARI
数值取整(3.1415926):3
日期格式化:2018-04-14
获取列表长度:3

遍历列表项:
  • 人:A0
  • 人:A1
  • 人:A2
遍历Map项key:
  • 键:emp5--值:雇员5
  • 键:emp4--值:雇员4
  • 键:emp3--值:雇员3
  • 键:emp2--值:雇员2
  • 键:emp1--值:雇员1
  • 键:emp0--值:雇员0
遍历Map项value:
  • 值:雇员5
  • 值:雇员4
  • 值:雇员3
  • 值:雇员2
  • 值:雇员1
  • 值:雇员0
引入其它ftl模板文件:
我是被include的模板内容
宏命令(自定义不带参数的标签)
不带参数宏:FreeMarker自动生成HTML
宏命令(自定义带参数的标签)
带参数宏:FreeMarker自动生成HTML--参数0--参数1--参数2
另一个ftl文件定义的变量

以上是使用FreeMarker模板引擎生成Html文件的全部过程。
下面列一下FreeMarker的基本语法(以下为转载勿喷):

数据类型
和java不同,FreeMarker不需要定义变量的类型,直接赋值即可。
字符串: value = "xxxx" 。如果有特殊字符 string = r"xxxx" 。单引号和双引号是一样的。
数值:value = 1.2。数值可以直接等于,但是不能用科学计数法。
布尔值:true or false。
List集合:list = [1,2,3] ; list=[1..100] 表示 1 到 100 的集合,反之亦然。
Map集合:map = {"key" : "value" , "key2" : "value2"},key 必须是字符串哦!
实体类:和EL表达式差不多,直接点出来。

字符串操作
字符串连接:可以直接嵌套${"hello , ${name}"} ; 也可以用加号${"hello , " + name}

字符串截取:string[index]。index 可以是一个值,也可以是形如 0..2 表示下标从0开始,到下标为2结束。一共是三个数。

比较运算符
== (等于),!= (不等于),gt(大于),gte(大于或者等于),lt(小于),lte(小于或者等于)。不建议用 >,< 可能会报错!
一般和 if 配合使用

内建函数
FreeMarker 提供了一些内建函数来转换输出,其结构:变量?内建函数,这样就可以通过内建函数来转换输出变量。
html: 对字符串进行HTML编码;
cap_first: 使字符串第一个字母大写;
lower_case: 将字符串转成小写;
upper_case: 将字符串转成大写;
size: 获得集合中元素的个数;
int: 取得数字的整数部分。

变量空判断
!   指定缺失变量的默认值;一般配置变量输出使用
??  判断变量是否存在。一般配合if使用 <#if value??>

宏指令
可以理解为java的封装方法,供其他地方使用。宏指令也称为自定义指令,macro指令
语法很简单:<#macro val > 声明macro ; 使用macro <@val />

命名空间
可以理解为java的import语句,为避免变量重复。一个重要的规则就是:路径不应该包含大写字母,使用下划线_分隔词语,myName --> my_name
语法很简单:<#import "xxx.ftl" as val>

其他没有说明的语法是因为和java一样,没什么特别之处。所以没有列出来。

SpringMVC整合FreeMarker

  • 1.SpringMVC环境配置(这里就不展开了,不会的百度一下吧)
  • 2.Maven配置:
    
         org.freemarker
         freemarker
         2.3.20
     
     
          org.springframework
          spring-context-support
          4.1.4.RELEASE
      
  • 3.springmvc的配置文件:
    
      
      
          
      
      
          
          
        
    
  • 4.Controller 层
import org.springframework.stereotype.Controller;
import org.springframework.ui.Model;
import org.springframework.web.bind.annotation.RequestMapping;

@Controller
public class HelloFreeMarkerController {
    
    @RequestMapping("/helloFreeMarker")
    public String helloFreeMarker(Model model) {
        model.addAttribute("name","ITDragon博客");  
        return "helloFreeMarker";
    }

}
  • 5.最后是Freemarker文件
  
  
  
FreeMarker Web  
  
  
    

Hello ${name} !

小结

1 知道了FreeMarker是一块模版引擎,可以生产xml,html,java等文件

2 知道了FreeMarker文件提供占位符,java文件提供数据,通过FreeMarker模版引擎生产有数据的页面,文中是将数据放在Map中。web应用可以用setter/getter 方法

3 知道了FreeMarker语法中字符串的显示特殊字符,截取的操作。以及一些内置方法的使用

4 重点了解FreeMarker的空判断知识点。判断变量是否为空用 "??" ,如果变量为空设置默认值。如果不注意空问题,可能会出现黄色页面的提示哦!

5 FreeMarker的宏概念,命名空间,引入文件,给变量赋值,集合的遍历等。

6 Freemarker 整合SpringMVC。

你可能感兴趣的:(《FreeMarker教程-笔录》)