smarty3中对内置函数的修改比較大,加入了很多新的功能:变量声明。表达式,流程控制,函数。数组等。可是建议不要在模版中去使用过于复杂的逻辑,而是要尽量将一些程序设计逻辑写到PHP中,并在模版中採用很easy的语法就可以调用。通常仅仅在模版中进行一些如变量输出,流程推断及数组遍历等操作就可以。
1. 变量声明
在模版中声明变量或用来在模版运行时为模版变量赋值,这是在Smarty3中新增的功能。
使用{assign}
,在模版运行时为模版变量或数组元素赋值 和在赋值时使用一些表达式
{$var=...}
是{assign}
的简写版。当中有三个属性(var,value和scope)和一个选项标签(nocache),当中var和value为必须属性,分别用来设置要分配的值的变量名和分配的值。scope可选,指定分配的变量范围,能够指定parent,root和global三个值。
用来设定变量的有效范围。
1.1 使用方法例如以下:
{assign var='name' value='brophp'} {*为变量$name赋值上brophp的值*}
{assign "name" "brophp"} {*这是assign函数属性的简写*}
{$name='brophp'} {*assign函数的简写*}
{*定义数组*}
{assign var=foo value=[1,2,3]} {*为变量$foo赋上一个索引数组值*}
{assign var=foo value=['y'=>'yellow','b'=>'blue']} {*为变量$foo赋上一个关联数组值*}
{assign var=foo value=[1,[1,3],4]} {*嵌套声明多维数组*}
{assign var=foo value=$x+$y} {*能够在属性中使用变量*}
{assign var="foo" value="`$foo+$bar`"} {*能够在属性值的字符串中使用变量及表达式*}
{*短变量分配*}
{$foo = $bar+2} {*短变量分配置值的方式*}
{$foo = strlen($bar)} {*PHP函数在变量值中使用*}
{$foo = myfunct($x+$y)*3} {*作为函数參数*}
{$foo.bar=1} {*数组元素的赋值*}
{$foo.bar.baz=1} {*多维数组元素的赋值*}
{$foo[]=1} {*为数组加入新元素*}
{$foo[$x+3]} {*变量作为数组索引*}
{$foo={counter}+3} {*在标签里嵌套标签*}
{$foo="this is message {counter}"} {*在引號里使用标签*}
1.2 參数scope使用方法
在加载模版中可见被加载模版(即为include进来)的分配变量。在声明变量时能够通过加入scope属性,并通过三个值来为调用的模版指定变量范围。
比如:
{*bar仅仅能在加载模版中常见*}
{assign var="bar" value="value"}
{*用了scope=parent,能够在自己和加载它的模版中可见*}
{assign var="foo" value="something" scope=parent}
{*全局变量在全部模版中可见*}
{assign var=foo value="bar" scope="global"}
{*能够在当前树形结构的"根"中赋值i一个变量。该变量可在全部使用该树形结构的模版中可见*}
{assign var=foo value="bar" scope="root"}
1.3 append使用方法
能够使用smarty的内置{append}
函数。在模版运行期间建立或追加模版变量数组。
{append var='name' value='Tom'} {*相似于$name[]='Tom'*}
{append var='name' value='Bob' index='first'} {*相似于$name.first='Bob*}
{append var='name' value='Meyer' index='last' scope='parent'} {*相似于$name.last='Meyer*}
2. 流程控制
在smarty3中新增了{for}
和{while}
两个内置函数
2.1 在smarty模版中使用{if}函数处理分支结构
要尽量去使用备用词来取代条件修饰符,这样能够避免在模版中使用和HTML标记符号冲突。
Smarty模版中在使用这些修饰词时,它们必须和变量或常量用空格隔开。此外,在PHP标准代码中,必须把条件语句包围在小括号里,而在Smarty中小括号的使用则是可选的,括号主要是用来改变运算符号的优先级别。
一些常见的选择控制结构使用方法例如以下:
{if $name eq 'Fred'}
Welcome Sir.
{elseif $name eq 'Wilma'}
Welcome Ma'am.
{else}
Welcom,whatever you are.
{/if}
{if $name eq "Fred" or $name eq 'Wilma'}
...
{/if}
{if $name == 'Fred' || $name == 'Wilma'}
...
{/if}
{if $name=='Fred'||$name=='Wilma'} {*错误的语法,条件符号和变量要用空格隔开*}
...
{/if}
{if($amount < 0 or $amount > 1000) and $volume >= #minVolAmt#} {*同意使用圆括号*}
...
{/if}
{if count($var) gt 0} {*能够嵌入函数*}
...
{/if}
{if is_array($foo)} {*数组检查*}
...
{/if}
{if isset($foo)} {*是否为空值检查*}
...
{/if}
{if $var is even} {*測试值为偶数还是基数*}
...
{/if}
{if $var is odd}
...
{/if}
{if $var is not odd}
...
{/if}
{if $var is div by 4} {*測试var是否能被4整除*}
...
{/if}
{if $var is even by 2} {*測试发现var是偶数,2个为一组*}
...
{/if}
{if isset($name) && $name == 'Blog'} {*很多其它{if}样例*}
{*do somthing*}
{elseif $name == $foo}
{* do somthing *}
{/if}
{if is_array($foo) && count($foo) > 0}
{*do a foreach loop*}
{/if}
2.2 在Smarty模版中使用{for}函数处理循环结构
和PHP中的for函数有所区别。在模版中使用{for},{forelse}标签创建一个简单的循环。{for}是一个块函数。须要使用{/for}结束。当循环无迭代时运行{forelse}.支持下面不同的格式。
{for $var=$start to $end}...{/for} {*步长为1的简单循环*}
{for $var=$start to $end step $step}...{/for} {*其它步长循环*}
{for $var=$start to $end max=$val}...{/for} {*设置循环的最大次数*}
当中$var是在{for}函数中定义的一个索引变量。须要给一个初始值作为索引的開始,还须要通过’to’关键字指定一个索引结束的值或变量。
也能够通过”step”关键字设置循环的步长,以及通过max中属性设置最大的循环次数。常见的语法例如以下:
<ul>
{for $foo = 1 to 3} {*设置循环使用变量$foo从1到3。循环3次*}
<li>{$foo}</li> {*循环体中的内容被输出3次。$foo的值分别为从1到3*}
</ul> {*for循环的结束标记*}
<ul>
{$start=1}
{$end=10}
{for $foo=$start to $end step 2} {*通过step设置循环变量递增的步长,本例设置为2。每次循环递增两次*}
<li>{$foo}</li>
{/for}
</ul>
<ul>
{for $foo=3 to 1}
<li>{$foo}</li>
{forelse}{*循环条件从3到1不成立,会运行forelse语句*}
{forelse}
<li>循环条件不成立</li>
{/for}
</ul>
2.3 在Smarty模版中使用{while}函数处理循环结构
也是一个块函数,和{if}相似,只是if语句条件成立循环一次。而while则循环多次
比如:
{$foo = 10}
{while $foo gt 0} {*假设变量$foo的值大于10就运行循环体的内容*}
{$foo--}
}
{/while}
3. 声明和调用模版函数
除了能够通过插件机制为smarty模版扩充函数,在smarty3还能够直接使用内置函数{function}
在模版中创建函数。能像调用插件函数一样调用它们
{function}
标签必须包括模版函数名的name属性。该name标签名必须能够调用模版函数。
{function name=menu} {*在模版中声明一个名称为Menu的函数*}
函数体
}
{/function}
{function menu} {*同上,简写形式*}
函数体
{/function}
{menu} {*调用函数menu,和调用插件函数一样调用在模版中自己定义的函数*}
在调用函数时,能够通过加入属性为模版中声明的函数传递參数,而声明的函数中不用去接收,直接就能够以变量的形式訪问属性參数
{function name=menu} {*在模版中声明的menu函数。并不用去接收參数传递的值*}
<b>{$data}</b>
<h1>{$style}</h1>
{function}
{menu data=$val style="list"}
能够声明默认的局部变量
{function name=menu level=0 data=$num+1} {*在模版函数声明时,同一时候声明两个局部变量*}
<b>{$data}</b>
<b>{$level}</b>
{/function}
{menu level=55}
能够通过{call}函数来调用{function}标签定义的模版函数和插件函数
{call name=menu data=$menu} {*调用函数menu,并传递一个data參数*}
{call menu data=$menu} {*简写*}
4. 数组遍历
在Smarty3中。{foreach}
函数的机制全部又一次改写。,格式和PHP中的foreach语法基本一样。而{section}
的使用方法没有变化
{foreach}
能够做{section}
能做的全部事
4.1 在smarty模版中使用{foreach}函数遍历数组
{foreach $arrayvar as $itemvar}...{/foreach}
仅仅遍历数组变量$arrayvar中的值
或
{foreach $arrayvar as $keyvar=>$itemvar}...{/foreach}
遍历数组变量$arrayvar的下标和值
假设使用{foreachelse}
从句。当数组变量无值时运行
{foreach} 属性说明
属性名 | 描写叙述 |
---|---|
@index | 包括当前数组的下表。開始时为0 |
@iteration | 包括当前虚幻的迭代,总是以1開始,和index不同,每迭代一次值自己主动加1 |
@first | 当{foreach}循环第一个时first为真 |
@last | 当{foreach}迭代到最后时last为真 |
@show | 用于检測{foreach}循环是否无数据显示,show为布尔值 |
@total | 包括{foreach}循环的总数(整数),能够用在{foreach}里面或后面 |
当然smarty2中的$smarty.foreach.name.property
语法仍然受支持
比如:testSmarty3.php
<?php
require_once '../Smarty/Smarty.class.php';
$smarty = new Smarty;
$smarty->left_delimiter="<{";
$smarty->right_delimiter="}>";
$arr1 = array('first'=>'Turing');
$arr2 = array('china'=>'中国','英国','us'=>'美国','日本','法国','德国','俄罗斯');
$arr3 = array('中国','英国','美国','日本','法国','德国','俄罗斯');
$smarty->assign('nameList',$arr1);
$smarty->assign('str','hahah');
$smarty->assign('arr2',$arr2);
$smarty->assign('arr3',$arr3);
$smarty->display('testSmarty3.html');
testSmarty3.html
<h2>foreach使用</h2>
<h4>@iteration,@index,@first,@last,@total</h4>
$arr2 = array('china'=>'中国','英国','us'=>'美国','日本','法国',
'德国','俄罗斯','');
<table border="1" cellspacing="0">
<{foreach $arr2 as $k=>$v}>
<{if $v@first}>
<tr>
<td>@iteration</td>
<td>@index</td>
<td>$k</td>
<td>$v</td>
</tr>
<{/if}>
<tr>
<td><{$v@iteration}></td>
<td><{$v@index}></td>
<td><{$k}></td>
<td><{$v}></td>
</tr>
<{if $v@last}>
<tr><td colspan="4">总共循环<{$v@total}>次</td></tr>
<{/if}>
<{foreachelse}>
$arr2为空
<{/foreach}>
<{if $v@show}>
<tr>
<td colspan="4">数据来源:最暖一天</td>
</tr>
<{/if}>
</table>
4.2 在smarty模版中使用{section}函数遍历数组
{section}属性:
属性名 | 描写叙述 | 类型 | 默认值 |
---|---|---|---|
name | 指定该循环的名称,当须要section循环内输出变量时,必须在变量后加上中括号包括着name变量。为必要參数 | 字符串 | 无 |
loop | 决定循环次数的变量名称。与数组变量同名。必要參数 | 数组变量 | 无 |
start | 确定循环開始运行的索引位置.假设是负数,则从尾部算起,比如有七个元素,则start为-2时,是从数组索引为5開始算起,假设超过循环数组的限制,则会被自己主动调整为最接近的合法值 | 整型 | 1 |
step | 决定了循环的步长,假设为负数,则从后向前遍历 | 整型 | 1 |
max | 循环最大运行次数 | 整型 | 数组长度 |
show | 决定是否显示该循环。能够使用这个參数进行调试 | 布尔类型 | true |
{section}
仅仅能循环下标从0開始的索引数组。能够使用{sectionelse}
,当loop属性为空时输出该区域内容。
在section中通过例如以下方式调用循环中的变量:
$smarty.section.sectionname.varname
当中{$smarty.section}
是Smarty的保留变量,sectionname即在section标记中指定的name属性值。而varname则是在section中被调用的特定变量名称。
section循环区域中能够调用的变量
变量名 | 描写叙述 |
---|---|
index | 用于显示当前循环的索引 |
index_prev | 用于显示上一个循环索引值。循环開始时。此值为-1 |
index_next | 用于显示下一个循环索引值。循环运行到最后一次时,此值仍然比当前索引值大1 |
iteration | 显示当前循环次数 |
first | 当前循环在第一次时为true |
last | 当前循环在最后一次时为false |
rownum | iteration别名 |
loop | 用于显示该循环上一次循环时的索引值。该值能够用于循环内部或循环结束后 |
show | true或false.假设设置为false,则该循环将不显示。假设指定了sectionelse子句。则该子句是否显示也取决于改值 |
total | 显示循环总数。能够在循环中使用,也能够结束后使用 |
将上面{foreach}
样例改写:
<h2>section使用</h2>
<h4>@iteration,@index,@first,@last,@total</h4>
$arr3 = array('中国','英国','美国','日本','法国',
'德国','俄罗斯','');
<table border="1" cellspacing="0">
<{section loop=$arr3 name='ls'}>
<{if $smarty.section.ls.first}>
<tr>
<td>@iteration</td>
<td>@index</td>
<td>下标</td>
<td>值</td>
</tr>
<{/if}>
<tr>
<td><{$smarty.section.ls.iteration}></td>
<td><{$smarty.section.ls.index}></td>
<td><{$smarty.section.ls.index}></td>
<td><{$arr3[ls]}></td>
</tr>
<{if $smarty.section.ls.last}>
<tr><td colspan="4">总共循环<{$smarty.section.ls.total}>次</td></tr>
<{/if}>
<{sectionelse}>
$arr2为空
<{/section}>
<{if $smarty.section.ls.show}>
<tr>
<td colspan="4">数据来源:最暖一天</td>
</tr>
<{/if}>
</table>
5. Smarty提供的其它内置函数
{block}
,{nocache}
将在后面的博客中提到。smarty2中的
{php}
和{include_php}
在模版中插入PHP程序。和模版使用规则相违背,已经在Smarty3中弃用。{strip},{literal},{ldelim},{rdelim}
和{insert}
都有其它更好的方法取代,前面的博客已经部分提到**{include}**
在多个模版中有同样的内容要输出。则能够将同样的部分在独立的模版中定义。然后在须要的模版中进行导入。在模版中使用{include}
在档期看模版中包括其它模版,必须使用file属性指明模版资源位置。
比如:头部文件header.tpl和尾部文件爱你footer.tpl都是独立的模版文件,希望每一个模版中都导入这两个文件。则:
{include file='header.tpl'} {*导入头部模版文件*}
这是当前模版中的主体内容部分
{include file='footer.tpl'} {*导入尾部模版文件*}
include有两个比較有用的特性:
- 能够在
{include}
标记中传入可选的assign属性,将导入的子模版内容不在当前模版中输出,而是赋给由assign属性指定的变量
{include file='header.tpl' assign="header"} {*模版header.tpl中的全部内容都赋给变量header*}
- 能够在导入子模版的同一时候向其传递各种属性。
以此方式传递给子模版的不论什么属性。仅仅能在被导入的文件里使用,不能用于模版的其它位置。假设传递的属性名在被包括模版中有同名变量,则该变量被传递的属性替代。
- 能够在导入子模版的同一时候向其传递各种属性。
{include file='header.tpl' title="main menu" table_bgcolor="#ccc"} {*在包括header.tpl时将这两个属性传递给它*}
这是当前模版文件里的主体部分
{include file="footer.tpl" logo="logo.gif"}