??xml version="1.0" encoding="utf-8" standalone="yes"?>亚洲国产成人影院播放,国产亚洲精品美女,亚洲国产中文在线视频http://www.tkk7.com/caizh2009/category/45999.html与大家共同成?/description>zh-cnFri, 20 Aug 2010 10:00:06 GMTFri, 20 Aug 2010 10:00:06 GMT60Velocity模板(VM)语言介绍http://www.tkk7.com/caizh2009/archive/2010/08/20/329495.html菜毛毛菜毛毛Fri, 20 Aug 2010 09:57:00 GMThttp://www.tkk7.com/caizh2009/archive/2010/08/20/329495.htmlhttp://www.tkk7.com/caizh2009/comments/329495.htmlhttp://www.tkk7.com/caizh2009/archive/2010/08/20/329495.html#Feedback0http://www.tkk7.com/caizh2009/comments/commentRss/329495.htmlhttp://www.tkk7.com/caizh2009/services/trackbacks/329495.html
Velocity是一个基于java的模板引擎(template engineQ。它允许MZ仅简单的使用模板语言Qtemplate languageQ来引用由java代码定义的对象?
当Velocity应用于web开发时Q界面设计h员可以和javaE序开发h员同步开发一个遵循MVC架构的web站点Q也是_面设计?
员可以只x面的显C效果,而由javaE序开发h员关注业务逻辑~码。Velocityjava代码从web面中分d来,q样为web站点的长
期维护提供了便利Q同时也为我们在JSP和PHP之外又提供了一U可选的Ҏ(gu)?
Velocity的能力远不止web站点开发这个领域,例如Q它可以从模板(templateQ生SQL和PostScript、XMLQ它?
可以被当作一个独立工h产生源代码和报告Q或者作为其他系l的集成lg使用。Velocity也可以ؓ(f)Turbine
web开发架构提供模板服务(template
serviceQ。Velocity+Turbine提供一个模板服务的方式允许一个web应用以一个真正的MVC模型q行开发?
Velocity能ؓ(f)我们作什么?
The Mud Store Example
假设你是一家专门出售Mud的在U商店的面设计人员Q让我们暂且U它?#8220;在线MUD商店”。你们的业务很旺Q客户下了各U类型和数量的mud?
单。他们都是通过输入用户名和密码后才登陆C的网站,登陆后就允许他们查看订单q购买更多的mud。现在,一U非常流行的mud正在打折销售。另外有一
些客戯律性的购买另外一U也在打折但是不是很行的Bright Red
MudQ由于购买的人ƈ不多所以它被安|在面的边~。所有用L(fng)信息都是被跟tƈ存放于数据库中的Q所以某天有一个问题可能会(x)冒出来:(x)Z么不使用
velocity来用户更好的浏览他们感兴趣的商品呢Q?
Velocity使得web面的客户化工作非常Ҏ(gu)。作Z个web site的设计h员,你希望每个用L(fng)陆时都拥有自q面?
你会(x)见了一些公司内的Y件工E师Q你发现他们每个人都同意客户应该拥有h个性化的信息。那让我们把软g工程师应该作的事情发在一边,看一看你应该作些什么吧?
你可能在面内嵌套如下的VTL声明Q?
<html>
<body>
Hello $customer.Name!
<table>
#foreach( $mud in $nudsOnSpecial )
#if ( $customer.hasPurchased( $mud ) )
<tr><td>$flogger.getPromo( $mud )</td></tr>
#end
#end
</table>
Velocity Template Language(VTL):AN introduction
VTL意味着提供最单、最Ҏ(gu)q且最整洁的方式合q面动态内宏V?
VTL使用references来在web
site内嵌套动态内容,一个变量就是一U类型的reference。变量是某种cd的refreenceQ它可以指向java代码中的定义Q或者从当前
面内定义的VTL statement得到倹{下面是一个VTL statement的例子,它可以被嵌套到HTML代码中:(x)
#set ( $a = “Velocity” )
和所有的VTL
statement一Pq个statement以#字符开始ƈ且包含一个directiveQset。当一个在U用戯求你的页面时QVelocity
Templating Engine查询整个页面以便发现所有#字符Q然后确定哪些是VTL statementQ哪些不需要VTL作Q何事情?
Q字W后紧跟一个directiveQsetӞq个set directive使用一个表辑ּQ用括号封闭)――一个方E式分配一个值给变量。变量被列在左边Q而它的D列在双Q最后他们之间用=号分剌Ӏ?
在上面的例子中,变量?aQ而它的值是Velocity。和其他的references一样以$字符开始,而值L以双引号闭。Velocity中仅有String可以被赋值给变量?
C以下的规则:(x)
使用$字符开始的references用于得到什么;使用#字符开始的directives用于作些什么?
Hello Velocity World!
一旦某个变量被分配了一个|那么你就可以在HTML文g的Q何地方引用它。在下面的例子中Q一个D分配l?foo变量Qƈ在其后被引用?
<html>
<body>
#set ( $foo = “Velocity” )
Hello $foo World!
</body>
</html>
上面的实现结果是在页面上打印“Hello Velocity WorldQ?#8221;
Z使包含VTL directives的statement更具有可L,我们鼓励你在新行开始每个VTL statementQ尽你不是必须q么作。Set directive在后面详细描述?
注释
单行注释Q?
## This is a single line comment.
多行注释Q?
#*
Thus begins a multi-line comment. Online visitors won’t
see this text because the Velocity Templating Engine will
ignore it.
*#
文档格式Q?
#**
This is a VTL comment block and
may be used to store such information
as the document author and versioning
information:
@version 5
@author
*#
References
在VTL中有三种cd的referencesQ变?variables)、属?properties)、方?methods)。作Z个
用VTL的页面设计者,你和你的工程师必dreferences的名U达成共识,以便你可以在你的template中用它们?
Everything coming to and from a reference被作Z个String对象处理。如果有一个对?foo是一个Integer对象Q那么Velocity调用它的toString()Ҏ(gu)这个对象{型ؓ(f)Stringcd?
变量
格式要求同java?
属?
例子Q?
$customer.Address
$purchase.Total
$customer.Address有两U含义。它可以表示Q查找hashtable对象customer中以Address为关键字的|也可
以表C用customer对象的getAddress()Ҏ(gu)。当你的面被请求时QVelocity确定以上两U方式选用那种Q然后返回适当的倹{?
Ҏ(gu)
一个方法就是被定义在java中的一D代码,q且它有完成某些有用工作的能力,例如一个执行计和判断条g是否成立、满等。方法是一个由$开始ƈ跟随VTL标识W组成的ReferencesQ一般还包括一个VTLҎ(gu)体。例如:(x)
$customer.getAddress()
$purchase.getTotal()
$page.setTitle( “My Home Page” )
$person.setAttributes( [“Strange”, “Weird”, “Excited”] )
前两个例?customer.getAddress()?purchase.getTotal()看v来挺想上面的属?customer.Address ?$purchase.Total。如果你觉得他们之间有某U联pȝ话,那你是正的?
VTL属性可以作为VTLҎ(gu)的羃写?customer.Address属性和使用$customer.getAddress()Ҏ(gu)h相同的效果。如果可能的话用属性的方式是比较合理的。属性和Ҏ(gu)的不同点在于你能够给一个方法指定一个参数列表?
正式reference标记
reference的正是格式如下:(x)
${mudSlinger} 变量
${customer.Address} 属?
${purchase.getTotal()} Ҏ(gu)
非正是格式更见常用,但是有时q是使用正是格式比较适合。例如:(x)你希望通过一个变?vice来动态的l织一个字W串?
Jack is a $vicemaniac.
本来变量?vice现在却变成了$vicemaniacQ这样Veloctiy׃知道(zhn)到底要什么了。所以,应该使用正是格式书写
Jack is a ${vice}maniac
现在Velocity知道变量?vice而不?vicemaniac?
Quiet reference notation
例如Q?
<input type=”text” name=”email” value=”$email” />
当页面的form被初始加载时Q变?emailq没有|q时你肯定是希望它能够显CZ个blank text来代替输?#8221;$email”q样的字Dc那么用quiet reference notation比较合适?
<input type=”text” name=”email” value=”$!email”/>
q样文本框的初始值就不会(x)是email而是Ig?
正式和quiet格式的reference notation也可一同用,像下面这P(x)
<input type=”text” name=”email” value=”$!{email}”/>
Getting literal
Velocity使用Ҏ(gu)字符$?来帮助它工作Q所以如果要在template里用这些特D字W要格外心。本节将讨论$字符?
货币字符
在VTL中?2.5q样的货币标识是没有问题得的QVTL不会(x)它错认为是一个referenceQ因为VTL中的referenceL以一个大写或者小写的字母开始?
Escaping valid VTL reference
VTL中?#8220;"”作ؓ(f)逃逸符?
例如Q?
#set( $email = “foo” )
$email
"$email
""$email
"""$email
render为:(x)
foo
$email
"foo
""$email
如果email变量没有被定义则
$email
"$email
""$email
"""$email
被render为:(x)
$email
"$email
""$email
"""$email
注意QVTL中未被定义的变量被认ؓ(f)是一个字W串Q所以以下例子:(x)
#set( $foo = “gibbous” )
$moon = $foo
的输出结果是Q?
$moon = gibbous
Case substitution
现在你已l对reference比较熟?zhn)了,你可以将他们高效的应用于你的template了。Velocity利用了很多java规范以方便了设计人员的用。例如:(x)
$foo
$foo.getBar()
## is the same as
$foo.Bar
$data.getUser(“jon”)
## is the same as
$data.User(“jon”)
$data.getRequest().getServerName()
# is the same as
$data.Request.ServerName
## is the same as
${data.Request.ServerName}
但是Q注意VTL中不?x)将reference解释为对象的实例变量。例如:(x)$foo.Name被解释为Foo对象的getNameQ)Ҏ(gu)Q而不是Foo对象的Name实例变量?
Directives
Reference允许设计者用动态的内容Q而directive使得你可以应用java代码来控制你的显C逻辑Q从而达C所期望的显C效果?
#set
#set directive被用于设|一个reference的倹{例如:(x)
#set ( $primate = “monkey” )
#set ( $customer.Behavior = $primate )
赋值左侧的QLHSQ必L一个变量或者属性reference。右侧(RHSQ可以是以下cd中一U:(x)
l 变量reference
l String literal
l 属性reference
l Ҏ(gu)reference
l number literal
l ArrayList
下面是应用各U类型的RHS的例子:(x)
Qset ( $monkey = $bill ) ##变量reference
Qset ( $monkey.Friend = “monica” ) ##String literal
Qset ( $monkey.Blame = $whitehouse.Leak )##属性reference
Qset ( $monkey.Plan = $spindoctor.weave($web) )##Ҏ(gu)reference
Qset ( $monkey.Number = 123 )##Number literal
Qset ( $monkey.Say = [“Not”, $my, “fault”] )##ArrayList
注意Q最后一个例子的取值方法ؓ(f)Q?monkey.Say.get(0)
RHS也可以是一个简单的术表达式:(x)
#set ( $value = $foo + 1 )
#set ( $value = $bar -1 )
#set ( $value = $foo * $bar )
#set ( $value = $foo / $bar )
如果你的RHS是一个nullQVTL的处理将比较Ҏ(gu)Q它?yu)指向一个已l存在的referenceQ这对初学者来讲可能是比较费解的。例如:(x)
#set ( $resut = $query.criteria(“name”) )
The result of the first query is $result
#set ( $resut = $query.criteria(“address”) )
The result of the second query is $result
如果$query.criteria(“name”)q回一?#8220;bill”Q?query.criteria(“address”)q回的是nullQ则昄的结果如下:(x)
The result of the first query is bill
The result of the first query is bill
看看下面的例子:(x)
#set( $criteria = ["name", "address"] )
#foreach( $criterion in $criteria )
#set( $result = $query.criteria($criterion) )
#if( $result )
Query was successful
#end
#end
在上面的例子中,E序不能智能的Ҏ(gu)$result的值决定查询是否成功。在$result?set后(added to the
contextQ,它不能被讄回nullQremoved from the
contextQ。打印的l果显CZơ查询结果都成功了,但是实际上有一个查询是p|的?
Z解决以上问题我们可以通过预先定义的方式:(x)
#set( $criteria = [“name”, “address”] )
#foreach( $criterion in $criteria )
#set( $result = false )
#set( $result = $query.criteria( $criterion ) )
#if( $result )
Query was successful
#end
#end
String Literals
当你使用#set directiveQString literal闭在一对双引号内?
#set ( $directoryRoot = “www” )
#set ( $templateName = “index.vm” )
#set ( $template = “$directoryRoot/$tempateName” )
$template
上面q段代码的输出结果ؓ(f)Qwww/index.vm
但是Q当string literal被封装在单引号内Ӟ它将不被解析Q?
#set ( $foo = “bar” )
$foo
#set ( $blargh = ‘$foo’ )
l果Q?
bar
$foo
上面q个Ҏ(gu)可以通过修改velocity.properties文g的stringliterals.interpolate = false的值来改变上面的特性是否有效?
条g语句
if/elseif/else
当一个web面被生成时使用Velocity?if directriveQ如果条件成立的话可以在面内嵌入文字。例如:(x)
#if ( $foo )
<strong>Velocity!</strong>
#end
上例中的条g语句在以下两种条g下成立:(x)
l $foo是一个boolean型的变量Q且它的gؓ(f)true
l $foo变量的g为null
q里需要注意一点:(x)Velocity context仅仅能够包含对象Q所以当我们?#8220;boolean”时实际上代表的时一个Boolean对象。即便某个方法返回的是一个boolean|Velocity也会(x)利用内省机制它转换Z个Boolean的相同倹{?
如果条g成立Q那?if?end之间的内容将被显C?
#elseif?else元素可以?if一同用。例如:(x)
#if( $foo < 10 )
<strong> Go North </strong>
#elseif( $foo == 10 )
<strong> Go East </strong>
#elseif( $foo == 6 )
<strong> Go South </strong>
#else
<strong> Go West </strong>
#end
注意q里的Velocity的数字是作ؓ(f)Integer来比较的――其他类型的对象得条件ؓ(f)falseQ但是与java不同它?#8220;Q=”来比较两个|而且velocity要求{号两边的值类型相同?
关系、逻辑q算W?
Velocity中用等h作符判断两个变量的关pR例如:(x)
#set ( $foo = “deoxyribonucleic acid” )
#set ( $bar = “ribonucleic acid” )
#if ( $foo == $foo )
In this case it’s clear they aren’t equivalent.So…
#else
They are not equivalent and this will be the output.
#end
Velocity有AND、OR和NOT逻辑q算W。下面是一些例子:(x)
## logical AND
#if( $foo && $bar )
<strong> This AND that </strong>
#end
## logical OR
#if ( $foo || $bar )
<strong>This OR That </strong>
#end
##logical NOT
#if ( !$foo )
<strong> NOT that </strong>
#end
循环
Foreach循环
例子Q?
<ul>
#foreach ( $product in $allProducts )
<li> $product </li>
#end
</ul>
每次循环$allProducts中的一个值都?x)赋l?product变量?
$allProducts可以是一个Vector、Hashtable或者Array。分配给$product的值是一个java对象Qƈ且可?
通过变量被引用。例如:(x)如果$product是一个java的Productc,q且q个产品的名字可以通过调用他的getNameQ)Ҏ(gu)得到?
现在我们假设$allProducts是一个HashtableQ如果你希望得到它的key应该像下面这P(x)
<ul>
#foreach ( $key in $allProducts.keySet() )
<li>Key: $key -> Value: $allProducts.get($key) </li>
#end
</ul>
Velocityq特别提供了得到循环ơ数的方法,以便你可以像下面q样作:(x)
<table>
#foreach ( $customer in $customerList )
<tr><td>$velocityCount</td><td>$customer.Name</td></tr>
#end
</table>
$velocityCount变量的名字是Velocity默认的名字,你也可以通过修改velocity.properties文g来改变它?
默认情况下,计数?#8220;1”开始,但是你可以在velocity.properties讄它是?#8220;1”q是?#8220;0”开始。下面就是文件中的配|:(x)
# Default name of loop counter
# variable reference
directive.foreach.counter.name = velocityCount
# Default starting value of the loop
# counter variable reference
directive.foreach.counter.initial.value = 1
Velocimacro properties
Velocity.properties文g中的某几行能够Velocimacros的实现更加灵zR注意更多的内容可以看Developer Guide?
Velocity.properties文g中的velocimacro.libraaryQ一个以逗号分隔的模板库列表。默认情况下Qvelocity查找唯一的一个库QVM_global_library.vm。你可以通过配置q个属性来指定自己的模板库?
Velocity.properties文g中的velocimacro.permissions.allow.inline属性:(x)有两个可选的
值true或者f(xi)alseQ通过它可以确定Velocimacros是否可以被定义在regular
template内。默认值是ture――允许设计者在他们自己的模板中定义Velocimacros?
Velocity.properties文g中的
velocimacro.permissions.allow.inline.replace.global属性有两个可选值true?
falseQ这个属性允怋用者确定inline的Velocimacro定义是否可以替代全局Velocimacro定义Q比如在
velocimacro.library属性中指定的文件内定义的VelocimacroQ。默认情况下Q此gؓ(f)false。这样就L本地
Velocimacro定义覆盖全局定义?
Velocity.properties文g中的
velocimacro.permissions.allow.inline.local.scale属性也是有true和false两个可?
|默认是false。它的作用是用于定你inline定义的Velocimacros是否仅仅在被定义的template内可见。换句话_如果q个
属性设|ؓ(f)trueQ一个inline定义的Velocimacros只能在定义它的template内用。你可以使用此设|实C个奇妙的VM?
门:(x)a template can define a private implementation of the second VM that
will be called by the first VM when invoked by that template. All other
templates are unaffected?
Velocity.properties文g中的velocimacro.context.localscope属性有true和false两个
可选|默认gؓ(f)false。当讄为trueӞM在Velocimacro内通过#set()对context的修改被认ؓ(f)是针Ҏ(gu)
velocimacro的本地设|,而不?x)永久的影响内容?
Velocity.properties文g中的velocimacro.library.autoreload属性控制Velocimacro
库的自动加蝲。默认是false。当讄为tureӞ对于一个Velocimacro的调用将自动查原始库是否发生了变化,如果变化重新加载它。这
个属性得你可以不用重新启动servlet容器而达到重新加载的效果Q就像你使用regular模板一栗这个属性可以用的前提是resource
loader~存是off状态(file.resource.loader.cache = falseQ。注意这个属性实际上是针对开发而非产品的?
Velocimacro Trivia
Velocimacro必须被定义在他们被用之前。也是_你的#macro()声明应该出现在用Velocimacros之前?
特别要注意的是,如果你试?parse()一个包?macro()的模ѝ因?parse()发生在运行期Q但是解析器?
parsetiem军_一个看似VM元素的元素是否是一个VM元素Q这?parse()-ing一lVM声明不按照预期的样子工作。ؓ(f)了得到预期的l?
果,只需要你单的使用velocimacro.library使得Velocity在启动时加蝲你的VMs?
Escaping VTL directives
VTL directives can be escaped with “"”P使用方式跟VTL的reference使用逃逸符的格式差不多?
## #include( “a.txt” ) renders as <ontents of a.txt>(注释?
#include( “a.txt” )