前阶段,有个项目需要使用php,于是组了一个3人组团队进行开发。
在项目中使用了模板解析,首先申明一下,个人认为:
1、模板解析并不是只有cms才用,中小型企业或者政府应用软件 尤其OA这样的自动化系统也是非常适用的。
2、如果运用于网站 则更主要的是模板解析速度和性能,如果应用于软件,更重要的是方便、可复用性强、可移植性强(这点尤其重要)
3、模板解析说到底就是变量替换,然而网页不比固定的软件界面,会产生的各种情况太多太复杂,因此不建议使用国外的很复杂的模板引擎,虽说开源但里面的代码非高手很难读懂,就算读得懂,也没有这么多时间去读和修改。(国内程序员大部分人都“没时间”,你懂得!!!)
4、模板解析过程中往往需要一些自定义函数,国外的如果要修改主程序比较复杂,依然是没时间。
因此让团队成员花了点时间开发了这个模板解析程序,我试了一下效果还可以,性能一般吧,不过我们是用在中小型企业应用里面的.
企业应用比较繁琐的就是表单,而且领导对表单的要求比较高,口味也很重,同样一个表的操作,新增和修改界面就要不一样,因此使用模板来加载数据是首选.
该模板解析程序功能比较简单 在后台进行数据绑定后,前台可以无限嵌套循环数据 ,个人比较喜欢的是,支持代码插件,譬如我想在页面中支持if else 判断,便可以自己写个解析程序加载进去,进行解析,同时在还支持了 自定义函数,函数的编写直接使用php代码即可。 以下我做一个说明
首先页面模板如下:
<!--open_func--> 这个代表开启函数支持,注意开启后 性能肯定要下降一点, <!--import_func="./tplfun.inc"--> //代表加载一些函数库,当然可以加载n个 <!--import_plug="./plug_if.inc"--> //代表加载了一个插件 ,我手写了一个if else 判断的插件,用来解析 if else 判断 <html> <head> <!-- BEGIN commonHtml --> //这个代表一个区域块,后台必须有个array 绑定这个区域块 <plugin:IF> //这个是一个插件 <!--if({miniui}==3)--> 设置成了红色 <!--if("{default}"=="默认变量")--> 设置了默认变量 <!--endif--> <!--else--> 没有设置成红色 <!--endif--> </plugin:IF> <!--include_file="./test.txt"--> // 这里支持 include 一个文件,如果该文件中含有模板解析标记 一样会被解析 <!-- END commonHtml --> //必须要有 结束标记 <style> .table th{text-align:center;} .table td{ line-height:25pt;height:25pt } </style> </head> <body> <plugin:IF> <!--if("{username}"=="admin888")--> 用户名正确 <!--else--> 用户名不正确 <!--endif--> </plugin:IF> 我是一个{setcolor(upper(username),"red")} <div id="main"> <div style="padding-left:11px;padding-bottom:5px;"> <!-- BEGIN commonHtml -->{setcolor(miniui,"red")}<!-- END commonHtml -->aa <!-- BEGIN datalist1 -->{setcolor(password,"red")}<!-- END datalist1 -->bb </div> <script> </body> </html>
这里注意,变量使用大括号 来表示 {username} 代表一个key是username的变量
然后需要有个 主程序来运行这个模板
代码如下:
require("./navtpl.inc"); $tpl=new navtemplate("./","test.tpl"); $data=array(); $data[]=array("default"=>"默认变量","miniui"=>"3"); $tpl->initTpl(); $data2=array(); $data2[]=array("password"=>"123"); $tpl->set_var("username",'admin888'); //设置一个变量 $tpl->set_block_var("commonHtml",$data);//设置区域块的数据源 $tpl->set_block_var("datalist1",$data2);//设置区域块的数据源 $tpl->parse(); //输出模板 结束战斗
由于开启了函数库 ,因此 我自己写了一个函数文件 来测试 如下
<?php function len($str) { return ""; } function lower($str) { return strtolower($str); } function upper($str) { return strtoupper($str); } function substring($str,$start,$length,$c="utf-8") { return mb_substr($str,$start,$length,$c); } function setcolor($st,$color) { return "<span style='color:$color'>$st</span>"; } ?>
下面是 一个 if else endif 的 语法解析, 这个插件 我写了一下,加载到了主模板程序,发现还是可以用 注意:只支持if else 不支持elseif 因为没有时间,我就没写elseif的支持
有兴趣的朋友 可以自行扩展
代码如下:
<?php class plugin_IF_ //插件类 { var $tag="/\<plugin:IF\>(.*?)\<\/plugin:IF\>/s"; function getLineCnt($begin,$end,$else,$list,$istrue) { if($begin==$end || $begin>$end || $begin<0) return "";//标签必须分行写 否则返回空 $ret=""; for($i=0;$i<=count($list)-1;$i++) { if($istrue) { $tmp=$list[$i]; if($else>0) { if($i>$else && $i<$end) continue; } if($i==$begin) { $tmp=preg_replace("/<!--if\((.*?)\)-->/s","",$tmp); } if($i==$end) { $tmp=preg_replace("/<!--endif-->/s","",$tmp); } if($i==$else) $tmp=preg_replace("/<!--else-->/s","",$tmp); if(trim($tmp)!="") $ret.=chr(13).$list[$i]; } else //非正确匹配 则要排除 begin 到end以外的 内容 { if($i<$begin || $i>$end || ($else>0 && $i>$else && $i<$end)) { $tmp=$list[$i]; if($i==$begin) $tmp=preg_replace("/<!--if\((.*?)\)-->/s","",$tmp); if($i==$end) $tmp=preg_replace("/<!--endif-->/s","",$tmp); if($i==$else) $tmp=preg_replace("/<!--else-->/s","",$tmp); if(trim($tmp)!="") $ret.=chr(13).$tmp; } } } return $ret; } function execFunc($cnt) { $list=explode(chr(13),$cnt); if(count($list)<2) return false;//必须有2行 否则 认为没有条件判读 $copy=array_reverse($list); $firstIF=""; $elseLinenum=0; $hasEndif=false; $beginLinenum=0; $endLinenum=0; $i=0; foreach($list as $line) { if(preg_match("/<!--if\((.*?)\)-->/s",$line,$match) && $firstIF=="")//代表匹配到第一个if 语句 { if($match && count($match)>1) { $firstIF=trim($match[1]); $beginLinenum=$i; } } if(preg_match("/<!--else-->/s",$line,$match) && $elseLinenum==0)//代表匹配到第一个else 语句 { $elseLinenum=$i; } if($firstIF!="" && $elseLinenum>0) break;//节省循环次数 $i++; } if(trim($firstIF)=="") return false;//没有开始标记 则认为没有代码 $i=count($list)-1; foreach($copy as $line) { if(preg_match("/<!--endif-->/s",$line))//代表匹配到最后一个if 语句 { $hasEndif=true; $endLinenum=$i; break; } $i--; } if(!$hasEndif) return false; //标签写错 则认为没有if判断 try { eval("\$ifvalue=(".$firstIF.");"); if(isset($ifvalue)) { if($ifvalue)//判断为true 则要获取所有内容 { return $this->getLineCnt($beginLinenum,$endLinenum,$elseLinenum,$list,true); } else return $this->getLineCnt($beginLinenum,$endLinenum,$elseLinenum,$list,false); } else return "";//条件判断写错 会 清除所有内容 } catch(Exception $ex) { return "";//代码发生未知错误 也会 清除所有内容 } } function getIfList($cnt) { preg_match_all($this->tag,$cnt,$match,PREG_SET_ORDER); if($match && count($match)>0) { foreach($match as $m) { $go=true; $i=0; $getresult=$m[1]; while($go) { if($i>=20) //最多支持20个标签 防止死循环 { $go=false;break; } $tmp=$this->execFunc($getresult); if($tmp===false) { $go=false;break; } $getresult=$tmp; $i++; } $cnt=preg_replace($this->tag,$getresult,$cnt,1); } } return $cnt; } function clearTag($cnt)//如果语法错误 则要去除冗余的 标记 { $cnt=preg_replace("/<!--if\((.*?)\)-->/s","",$cnt); $cnt=preg_replace("/<!--endif-->/s","",$cnt); return $cnt; } function parse($cnt)//插件必须拥有这个 方法 否则不会执行 插件任何内容 { $list=explode(chr(13),$cnt); $cnt=$this->getIfList($cnt); $cnt=$this->clearTag($cnt); return $cnt; } } ?>
这里插件 要注意的是 必须是一个 class 文件 并且 类名 必须遵循 plugin_类名_ 这样的形式 ,转换到模板 就是
<plugin:类名>....</plugin:类名> /// 这里大小写 是很敏感的。插件 必须拥有parse方法 返回 解析后的内容
个人认为的优点:
1、灵活,扩展性好
2、修改核心无障碍。
3、没有老外的那种 一大堆文件和代码的风格,虽说老外写的东西很牛,但是如果开发的是国人的软件 还是用国人的东西比较适合。
缺点不说了,bug肯定有 ,性能没有老外的好,高手会认为是垃圾,和smarty 不是同一个东西,勿喷
欢迎 各位有兴趣的 好朋友 提供更好的 语法插件
以下是一些注意点 :
1、使用上面程序 必须使用phh5 ,必须开启mb_string 必须支持eval
2、主程序是utf-8,因此必须使用utf8编码, 理论上 模板是gb2312也没问题,但是我没测试
3、由于项目还没交付,同时该模板程序还在不断完善和修改,目前是v0.1 ,因此模板主解析程序 我进行了压缩, 后面会放出 带注释版本的 纯原文件。
4.本模板特别适用于 一线开发人员使用(很忙 很没时间的开发人员), 不建议新手使用(新手还是先学好基础为好),不建议高手使用(高手可能会认为此物是垃圾)
测试程序 我放到了我的网盘 里面 http://url.cn/7dKBks (qq的微云)。
如果不能下载 请留下 邮箱