8. Smarty3:模版中的内置函数 1. 变量声明 2. 流程控制 3. 声明和调用模版函数 4. 数组遍历 5. Smarty提供的其它内置函数


smarty3中对内置函数的修改比較大,加入了很多新的功能:变量声明。表达式,流程控制,函数。数组等。可是建议不要在模版中去使用过于复杂的逻辑,而是要尽量将一些程序设计逻辑写到PHP中,并在模版中採用很easy的语法就可以调用。通常仅仅在模版中进行一些如变量输出,流程推断及数组遍历等操作就可以。


在模版中声明变量或用来在模版运行时为模版变量赋值,这是在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提供的其它内置函数

  1. {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有两个比較有用的特性:

  1. 能够在{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"}