FreeMaker简介

FreeMaker其实是一种比较简单的网页展示技术,说白了就是网页模板和数据模型的结合体。这种结合模式的好处就是,分离了网页界面设计人员和编程人员的工作,让他们各司其职。

据个人理解,FreeMaker大致的工作方式是,网页模板里面嵌入了数据模型中的数据、FreeMake自定义流程控制语言、FreeMake自定义的操作函数等等,在装载网页的时候,FreeMaker模板自动从数据模型中提取数据,并解释整个网页为我们熟知的HTML页面。

数据模型(The Data Model)

首先看看下面的两个例子(注:这里只是用示意图的形式来展示数据模型的概念,此处的图片并不代表源码中的一个文本性质的文件):

这种结构有点类似于目录树结构,其中,String性质的属性值我们都用双引号或单引号围起来,如"mouse" or 'mouse'表示String类型。同理,Number数字则不需要任何引号,如字符"50"和50分别是两种不同性质的节点。

在这里,有以下几个FreeMaker术语需要了解:

●hashes:处于“目录”位置的节点。如左图中的root, animals, mouse,
elephant, python, whatnot;

●scalars:叶子节点,即处于“文件”位置的节点。如左图中的size, price, test, because,如果你要访问price节点,我们可以通过如下的方式在HTML代码中嵌入${animals.mouse.price},scalars可以有多种类型的值。

●sequences:类似于hashes,但是它不用名字来命名子节点,而是把该子节点写作为一个数字序列,相关的scalars置于这些数字序列下。如右图中的animals和whatnot.fruits,如果你要访问第一个动物的name,那么可以写为${animals[0].name};访问第2个水果,则写为${whatnot.fruits[1]}。



网页模板(Template)



我们在简介中已经初步介绍了一下网页模板的概念。网页模板事实上就是一个静态的HTML文件,里面包含了以下几种元素:

●Interpolation:具体的格式为${expression},在数据模型中,我们已经提到了这个特殊的符号,主要还是用来容纳模板中节点内容的。此外,也可以采用#{expression}或#{expression; format}来展示数字(如#{x; m1M3}表示x小数部分最小长度为1最大长度为3,多出部分四舍五入)。

●FTL tags(FreeMarker Template Language tags):简单的说,就是FreeMaker的子定义流程控制语言和操作函数,他们均以HTML tag的形式存在,只不过采用了#和@的标志符来区别于普通的HTML标记。FTL tags 不能互相嵌套,如<#if <#include 'foo'>='bar'>...是一个错误的用法。

FreeMaker默认使用尖括号来涵盖FTL tags的内容,但是系统也可以统一使用方括号来修饰FTL tags,两者不能同时混用(注:每一个网页Template可以在最开始处使用<#ftl>或[#ftl]来标示整个模板的基本配置,因此后文中的所有的tags的括号都应该和ftl的一致,不一致的FTL tags一律视为普通的Text)。

●注释(Comments):类似于HTML的注释,表示方式为 <#-- and --> (注意:不是),嵌入在里面的所有内容均不会被FreeMaker本身所解释。此外,Comments可以处于FTL tags和Interpolation中间。

除了以上三种之外,其他的网页代码都不会被FreeMaker本身所解释,而最终保留并直接呈现出来,FreeMaker称之为Text。

表达式(Expressions)

1. 基础数据类型

String: 由前后引号括起来的字符串,如:"Green Mouse", "I am \"tailsherry\"!", r"C:\raw\string",和其他的编程语言一样,FreeMaker的String类型也支持转义字符:

备注:

●在转义符\x之后,应该是1至4个十六进制的数值,如:"\xA9
1999-2001", "\x0A9
1999-2001", "\x00A9
1999-2001"均表示内嵌一个Copy Right标志。如果\x之后超出4个数值,那么FreeMaker就成白痴了; :P

●另外,在String类型中,我们还有一个"生字符串"(Raw string literals)的概念。在生字符串中,\和${都失去FreeMaker本身的意义了,直接当作了普通的文本处理。表示生字符串的方式是,在第一个引号之前,加上一个字母r,如r"C:\foo\bar"输出就应该是"C:\foo\bar";

●如何引用字符串?我们常用的方式有${"Hello
${user}!"},此处假设user是tailsherry,那么输出结果为Hello tailsherry!。同样,你也可以直接用+连接已有的字符串${"Hello" + user + "!"},此结果和前者一样;

●如何取用一个字符串中的部分序列?可以用${user[0]},
${user[1..4]}, ${user[5..]}。当然,推荐的还是用内嵌函数substring来解决了${user?substring(0,0)}, ${user?substring(1,4)},
${user?substring(5)};

●一些正确的用法:〈h1〉Hello ${name}!〈/h1〉    <#include
"/footer/${company}.html">;

●一些典型的错误用法:<#if
${isBig}>Wow!    <#if "${isBig}">Wow!。

Number: 在模板中直接用数字表示,如:150 or -90.05 or +0.001。

备注:

●数字中只能用.号,不可以用其他的符号(如,)来分割为小数;

●你可以用+和-来标示该数字为正数和负数;

●到目前为止,Number还不支持科学计数法,如1E3是错误的用法。

Date: 分为Date, Time, DateTime(TimeStamp)三种格式,这里不讨论具体的Date的实现规则,这是一个比较复杂的问题;

Boolean: 布尔值,表示逻辑上的true/false,实际使用的时候请不要用引号把true/false涵括起来。

Sequences: 序列值,可以理解为一个java的枚举/集合类型。常用的形式为:["winter",
"spring", "summer", "autumn"]。

备注:

●序列中的子项目可以是一个任意的Expression,如[2 + 2, [1, 2, 3, 4], "whatnot"];

●你可以用区间的方式来表示一组连续的数字,如2..5 (注意了,这里的2..5没有前后方括号的),该序列即等同于[2, 3, 4, 5] (ps: 一般来说,前者更加有效率一点,因为它占用的内存小一点,访问速度更快);同样,你也可以倒过来这个序列使用5..2,或者干脆省略掉结束数5.., 5,6,7,... 表示从5到无穷大!

●可以合并两个Sequences序列,如["Joe", "Fred"] +
["Julia", "Kate"] as user,那么${user}的将是一个新的序列,其值为["Joe", "Fred",
"Julia", "Kate"];

●建议不要多次反复合并Sequences序列,这样降低系统对合并后序列的访问速度。

Hashes: 传统的键/值对方式,类似于java的Map,表示形式为{"name":"green mouse",
"price":150},其中[值]部分可以是一个任意的Expression。

备注:

●Hashes可以类似于Sequences一样进行两者合并,如{"Joe":23, "Fred":25} + {"Joe":30,
"Julia":18}。如果前后存在有两个相同的[键],那么合并后以+号右边的Hashes对应的[值]为基准;

●同样,建议不要多次反复合并Hashes,这样降低系统对合并后Hashes的访问速度。

2. 数学运算

类似于java的数学运算,加+减-乘*除/余%,没有什么比较特别的地方。

数学预算仅限于Number类型,如果尝试用String和Number进行运算,将会出现异常!当然,有一个例外,如果你使用${3 + "5"},那这里的数字会自动转变为String来处理,事实上这里的+就是一个字符串的连接符了,返回的结果应该就是字串"35"了。

3. 逻辑运算

类似于java的逻辑运算,=,==,!=,〉,〈,〉=,〈=,||,&&,!等等,许多人对这些符号都耳熟能详,就不一一介绍了。

有几点需要注意的地方:

●=和==在FreeMaker中都可以用来等值判断;

●!=,〉,〈,〉=,〈=仅用于Number和Date类型的比较,如果有String参与比较,将会抛出异常;

●由于〉,〈,〉=,〈=中的〈〉等同于tags中的前后封闭符,这里应该分别用>,<,>e, ●使用expression??,可以判断这个expression是否有指定值,如果有值,那么返回true,否则false。可以结合[5 默认值设定]部分理解。

4. 内嵌函数(Built-ins)

不同于一般的java和c的访问连接符,这里用?顶替.来处理。如${test?html}、${test?upper_case?html}等。

简单列举一下几个常见的内嵌函数:

●String字符串函数

html: 将特殊的HTML字符转为转义符号,常见的〈〉&等变化为<>&等。

cap_first:
将String的第一个字母大写。

lower_case:
将String的所有字母小写。

upper_case:
将String的所有字母大写。

trim: 去除String前后的空白符。

●Sequence序列函数

size: 标志Seuqence的大小。

●Number数值函数

int: 获得一个Number的整数部分的值,如-1.9?int返回-1。

5. 默认值设定(Default value)

FreeMaker用一个!符号表示某个expression的默认值,如${mouse!"No mouse."}。

有几点需要注意的地方:

●如果!后面的默认值没有内容,那么默认为一个空String或空Sequence或空Hash。注意,对于Number和Boolean如果让默认值为0和false,必须显式的指定默认值,不可以省略;

●如${product.color!"red"}只保证了最后一个color的默认值为red,如果product没有指定,同样会引发错误。但是, (product.color)!"red" 却一定保证这个值为red,不管product或color指不指定。

6. 操作符的优先级(Operator precedence)

所有FreeMaker操作符的优先级类似于java, C等高级语言,除了一些FreeMaker本身独有的操作符之外。

结束语:对于一个有思想的人来说,没有一个地方是荒凉偏僻的。在任何逆境中,他(她)都能充实和丰富自己。

可爱博主:AlanLee

博客地址:http://www.cnblogs.com/AlanLee

本文出自博客园,欢迎大家加入博客园。

FreeMaker开发教程的更多相关文章

  1. ASP.NET Aries 入门开发教程7:DataGrid的行操作(主键操作区)

    前言: 抓紧勤奋,再接再励,预计共10篇来结束这个系列. 上一篇介绍:ASP.NET Aries 入门开发教程6:列表数据表格的格式化处理及行内编辑 本篇介绍主键操作区相关内容. 1:什么时候有默认的 ...

  2. ASP.NET Aries 入门开发教程6:列表数据表格的格式化处理及行内编辑

    前言: 为了赶进度,周末也写文了! 前几篇讲完查询框和工具栏,这节讲表格数据相关的操作. 先看一下列表: 接下来我们有很多事情可以做. 1:格式化 - 键值的翻译 对于“启用”列,已经配置了格式化 # ...

  3. ASP.NET Aries 入门开发教程4:查询区的下拉配置

    背景: 今天去深圳溜达了一天,刚回来,看到首页都是微软大法好,看来离.NET的春天就差3个月了~~ 回到正题,这篇的教程讲解下拉配置. 查询区的下拉配置: 1:查询框怎么配置成下拉? 在配置表头:格式 ...

  4. Android快乐贪吃蛇游戏实战项目开发教程-01项目概述与目录

    一.项目简介 贪吃蛇是一个很经典的游戏,也很适合用来学习.本教程将和大家一起做一个Android版的贪吃蛇游戏. 我已经将做好的案例上传到了应用宝,无病毒.无广告,大家可以放心下载下来把玩一下.应用宝 ...

  5. Senparc.Weixin.MP SDK 微信公众平台开发教程(十八):Web代理功能

    在Senparc.Weixin.dll v4.5.7版本开始,我们提供了Web代理功能,以方便在受限制的局域网内的应用可以顺利调用接口. 有关的修改都在Senparc.Weixin/Utilities ...

  6. Senparc.Weixin.MP SDK 微信公众平台开发教程(十七):个性化菜单接口说明

    前不久微信上线了个性化菜单接口,Senparc.Weixin SDK也已经同步更新. 本次更新升级Senparc.Weixin.MP版本到v13.5.2,依赖Senparc.Weixin版本4.5.4 ...

  7. 微信公众账号 Senparc.Weixin.MP SDK 开发教程 索引

    Senparc.Weixin.MP SDK从一开始就坚持开源的状态,这个过程中得到了许多朋友的认可和支持. 目前SDK已经达到比较稳定的版本,这个过程中我觉得有必要整理一些思路和经验,和大家一起分享. ...

  8. Senparc.Weixin.MP SDK 微信公众平台开发教程(三):微信公众平台开发验证

    要对接微信公众平台的"开发模式",即对接到自己的网站程序,必须在注册成功之后(见Senparc.Weixin.MP SDK 微信公众平台开发教程(一):微信公众平台注册),等待官方 ...

  9. Senparc.Weixin.MP SDK 微信公众平台开发教程(四):Hello World

    =============  以下写于2013-07-20 ============= 这一篇文章其实可以写在很前面,不过我还是希望开发者们尽多地了解清楚原理之后再下手. 通过上一篇Senparc.W ...

随机推荐

  1. bzoj 3028: 食物 -- 母函数

    3028: 食物 Time Limit: 3 Sec  Memory Limit: 128 MB Description 明明这次又要出去旅游了,和上次不同的是,他这次要去宇宙探险! 我们暂且不讨论他 ...

  2. 【MySql】——MHA+GTID+failover+binlog-server+Atlas

    一.环境准备 1.mysql-db01 #系统版本 [root@mysql-db01 ~]# cat /etc/redhat-release CentOS release 6.7 (Final) #内 ...

  3. OC 中 @synthesize 关键字介绍和使用

    @synthesize用法 )@property int age; @synthesize age; 表示生成.h中变量 age的 get和 set方法 注意: 如果@synthesize 变量名要先 ...

  4. openresty源码剖析——lua代码的执行

    上一篇文章中我们讨论了openresty是如何加载lua代码的 那么加载完成之后的lua代码又是如何执行的呢 ##代码的执行  在init_by_lua等阶段  openresty是在主协程中通过lu ...

  5. TPshop手机新模板的用户消息实现

    今天在开发TPshop的手机新模板的消息页面,姑且记录一下. 首先,点击下方右上角,进入消息页面: 数据库中目前模拟了三条数据,有: 点击上图右上角,有: 这个 消息设置 功能是新增的,而且类型由原本 ...

  6. python 用户交互

    #coding=utf8 name = input("name:") age = int(input("age:")) job = input("jo ...

  7. 使用Redis实现分布式锁

    在天猫.京东.苏宁等等电商网站上有很多秒杀活动,例如在某一个时刻抢购一个原价1999现在秒杀价只要999的手机时,会迎来一个用户请求的高峰期,可能会有几十万几百万的并发量,来抢这个手机,在高并发的情形 ...

  8. java实现简单计算器

    首先利用字符串数组保存计算器上的按钮的标签名 private final String[] str = {"7","8","9"," ...

  9. Java IO流学习总结(2)

    写在前面:本文章基本覆盖了java IO的全部内容,java新IO没有涉及,因为我想和这个分开,以突出那个的重要性,新IO哪一篇文章还没有开始写,估计很快就能和大家见面.照旧,文章 依旧以例子为主,因 ...

  10. java日期工具类(Long型,Date型,yyyyMMdd型)等

    import java.sql.Timestamp; import java.text.ParsePosition; import java.text.SimpleDateFormat; import ...