PHP模板解析入门
学习:李炎恢PHP视频第二季
模板引擎的特点:
1.鼓励分离:让更个系统的可读性和维护性得到提高。
2.促进分工:使得程序员和美工去专心处理自己的设计。
3.比PHP更容易解析:编译文件和缓存文件加载更快、占资源更少。
4.增加安全性:可限制模板设计师进行不安全的操作的能力避免误删误访问等
一个PHP对应一个模板文件,经过调用模板类并且经过模板解析类之后,生成编译过后的php编译文件,为了加快访问速度,并且生成纯静态的HTML文件,作为缓存文件。
简易目录:
1.cache文件夹存放静态HTML文件,加快访问速度
2.config文件夹存放系统变量的XML文件
3.includes 存放模板解析类库
-------------Parser.class.php 模板文件解析类 用于解析tpl文件,将其解析成php文件
------------Templates.class.php 模板类 负责提供给index.php调用传递数据和调用模板文件
4.templates文件夹 存放tpl文件
5.Templates文件夹 存放解析过后的tpl文件,为PHP文件
6.Index.php 调用入口文件
7.template.inc.php 初始化模板文件
8.Test.php 测试文件类
一、定义初始化变量
template.inc.php
<?php
//设置utf-8编码
header('Content-Type:text/html;charset=utf-8');
//设置网站更目录
define('ROOT_PATH',dirname(__FILE__));
//设置模板文件目录
define('TPL_DIR',ROOT_PATH.'/templates/');
//编译文件目录
define('TPL_C_DIR',ROOT_PATH.'/templates_c/');
//缓存目录
define('CACHE',ROOT_PATH.'/cache/');
//是否开启缓冲区
define('IS_CACHE',true);
//开启缓冲区
IS_CACHE&&ob_start();
//引入模板类
require ROOT_PATH.'/includes/Templates.class.php';
//实例化模板类
$_tpl = new Templates();
二.入口文件index.php
引入初始化文件
require dirname(__FILE__).'/template.inc.php';
global $_tpl;
//声明一个变量
$_name = '周起';
$_array = array(1,2,3,4,5,6,7);
//调用方法注入变量给模板
$_tpl->assign('name',$_name);
$_tpl->assign('content','学习PHP');
$_tpl->assign('a',1);
$_tpl->assign('array',$_array);
//载入tpl文件
$_tpl->display('index.tpl');
三、Templates.class.php
<?php
class Templates{
//字段用于接收动态变量
private $_vars = array();
//保存系统变量
private $_config = array();
//创建构造方法验证各个目录是否存在
public function __construct()
{
if (!is_dir(TPL_DIR)||!is_dir(TPL_C_DIR)||!is_dir(CACHE)){
exit('ERROR:模板目录或编译目录或缓存目录不存在,请手动添加');
}
//解析xml系统变量
$_sxe = simplexml_load_file('config/profile.xml');
$_taglib = $_sxe->xpath('/root/taglib');
foreach ($_taglib as $tag){
$this->_config["$tag->name"] = $tag->value;
}
}
//用于注入变量
/**
* @param $_var模板变量名
* @param $_value模板变量值
*/
public function assign($_var,$_value)
{
if (isset($_var)||!empty($_var)
){
$this->_vars[$_var] = $_value;
}else{
exit('ERROR:请设置模板变量');
}
}
//display() 生成编译文件和缓存文件
/**
* @param $_file模板文件路径
*/
public function display($_file)
{
//设置模板路径
$_tplFile = TPL_DIR.$_file;
//判断模板文件是否存在
if (!file_exists($_tplFile)){
exit('ERROR:'.$_file.'文件不存在');
}
//生成编译文件路径
$_parFile = TPL_C_DIR.md5($_file).$_file.'.php';
//生成缓存文件
$_cacheFile = CACHE.md5($_file).$_file.'.html';
//当第二次相同文件的时候,直接载入缓存文件,避开编译
if (IS_CACHE){
//缓存文件和编译文件同时存在
if (file_exists($_cacheFile)&&file_exists($_parFile)){
//并且模板文件修改时间大于编译文件
if (filemtime($_cacheFile)>=filemtime($_parFile)&&filemtime($_parFile)>=filemtime($_tplFile)){
//直接载入静态文件
include $_cacheFile;
return ;
}
}
}
//当编译文件不存在或者模板文件被修改的时候 重新生成编译文件
if (!file_exists($_parFile ||filemtime($_parFile)<filemtime($_tplFile))){
//引入模板解析类
require ROOT_PATH.'/includes/Parser.class.php';
//实例化对象的时候将模板文件路径传递进去
$_parser = new Parser($_tplFile);
$_parser->compile($_parFile);
}
//载入编译文件
include $_parFile;
if (IS_CACHE){
//获取缓冲区数据放入缓存文件
file_put_contents($_cacheFile,ob_get_contents());
//清除缓冲区
ob_end_clean();
//载入缓存文件
include $_cacheFile;
}
}
}
当实例化Template对象的时候,会先判断各个目录是否存在
当index.php调用$_tpl->assign()方法的时候,调用Template对象中的assign方法,在Template中有一个私有属性$_var为数组类,用于存放assign中的参数
//用于注入变量
/**
* @param $_var模板变量名
* @param $_value模板变量值
*/
public function assign($_var,$_value)
{
if (isset($_var)||!empty($_var)
){
$this->_vars[$_var] = $_value;
}else{
exit('ERROR:请设置模板变量');
}
}
如果变量不存在,或者变量为空的话,提示设置模板变量,否则,以变量名为数组索引,变量值为相对索引的值,存放入$_var私有变量中。
如果出现相同变量名就会出现因为索引相同,出现覆盖的情况。
//声明一个变量
$_name = '周起';
//调用方法注入变量给模板
$_tpl->assign('name',$_name);
$_tpl->assign('content','学习PHP');
$_tpl->assign('content','不想成为码农');
die();
这个时候掉用assign()方法,assign()方法改写为
public function assign($_var,$_value)
{
if (isset($_var)||!empty($_var)){
$this->_vars[$_var] = $_value;
}else{
exit('ERROR:请设置模板变量');
}
print_r($this->_vars);
}
在网页端打开访问
当调用Template类中display()方法的时候,传递进来一个模板文件名称
$_tpl->display(‘index.tpl’);
/**
* @param $_file模板文件路径
*/
public function display($_file)
{
//设置模板路径
$_tplFile = TPL_DIR.$_file;
//判断模板文件是否存在
if (!file_exists($_tplFile)){
exit('ERROR:'.$_file.'文件不存在');
}
//生成编译文件路径
$_parFile = TPL_C_DIR.md5($_file).$_file.'.php';
//生成缓存文件
$_cacheFile = CACHE.md5($_file).$_file.'.html';
//当第二次相同文件的时候,直接载入缓存文件,避开编译
if (IS_CACHE){
//缓存文件和编译文件同时存在
if (file_exists($_cacheFile)&&file_exists($_parFile)){
//并且模板文件修改时间大于编译文件
if (filemtime($_cacheFile)>=filemtime($_parFile)&&filemtime($_parFile)>=filemtime($_tplFile)){
//直接载入静态文件
include $_cacheFile;
return ;
}
}
}
//当编译文件不存在或者模板文件被修改的时候 重新生成编译文件
if (!file_exists($_parFile ||filemtime($_parFile)<filemtime($_tplFile))){
//引入模板解析类
require ROOT_PATH.'/includes/Parser.class.php';
//实例化对象的时候将模板文件路径传递进去
$_parser = new Parser($_tplFile);
$_parser->compile($_parFile);
}
//载入编译文件
include $_parFile;
if (IS_CACHE){
//获取缓冲区数据放入缓存文件
file_put_contents($_cacheFile,ob_get_contents());
//清除缓冲区
ob_end_clean();
//载入缓存文件
include $_cacheFile;
}
}
1.接收到tpl模板名称,定位到该模板文件路径,并且检测模板文件是否存在。
2.并且同时声明两个变量用于存放模板编译文件路径和纯静态HTML缓存文件路径
3.模板文件和编译文件和静态文件规则是这样:
如果编译文件和缓存文件都不存在,直接载入编译文件,并且生成静态缓存文件
当编译文件不存在或者模板文件被修改的时候,重新生成编译文件,并且生成静态缓存文件
当第二次访问相同页面,并且编译文件和缓存文件同时存在,并且缓存文件的修改时间大于编译文件的时候,直接访问静态缓存文件
知识点:file_put_contents() 函数把一个字符串写入文件中。
缓存机制:http://www.cnblogs.com/usa007lhy/p/5421545.html(推荐)
filemtime() 函数返回文件内容上次的修改时间。
fileatime() 函数返回指定文件的上次访问时间。
当第一次或者模板文件发生改变的时候,重新解析模板文件,并且将其解析
//当编译文件不存在或者模板文件被修改的时候 重新生成编译文件
if (!file_exists($_parFile ||filemtime($_parFile)<filemtime($_tplFile))){
//引入模板解析类
require ROOT_PATH.'/includes/Parser.class.php';
//实例化对象的时候将模板文件路径传递进去
$_parser = new Parser($_tplFile);
$_parser->compile($_parFile);
}
<?php
class Parser{
//保存模板内容
private $_tpl;
//接收模板文件路径
public function __construct($_tplFile)
{
if (!$this->_tpl = file_get_contents($_tplFile)){
exit('ERROR:模板文件读取错误');
}
}
public function compile($_parFile)
{
$this->parConfig();
$this->parInclude();
$this->parForeach();
$this->parCommon();
$this->parIf();
$this->parVar();
//生成编译文件之前解析模板内容
if (!file_put_contents($_parFile,$this->_tpl)){
exit('ERROR:编译文件生成出错');
}
}
//解析if语句
private function parIf(){
$_pattenIf = '/\{if\s+\$([\w]+)\}/';
$_pattenEndIf = '/\{\/if\}/';
$_pattenElse = '/\{else\}/';
if (preg_match($_pattenIf,$this->_tpl)){
if (preg_match($_pattenEndIf,$this->_tpl)){
//匹配if开头
$this->_tpl = preg_replace($_pattenIf,'<?php if($this->_vars[\'$1\']){?>',$this->_tpl);
//匹配if结尾符并且更改为php形式
$this->_tpl = preg_replace($_pattenEndIf,"<?php };?>",$this->_tpl);
//匹配else
if (preg_match($_pattenElse,$this->_tpl)){
$this->_tpl = preg_replace($_pattenElse,"<?php }else{;?>",$this->_tpl);
}
}else{
exit('ERROR:IF语句没有关闭');
}
}
}
//解析PHP代码注释
private function parCommon(){
$_pattern = '/\{#\}(.*)\{#\}/';
if (preg_match($_pattern,$this->_tpl)){
$this->_tpl = preg_replace($_pattern,'<?php /*$1*/;?>',$this->_tpl);
}
}
//解析普通变量
private function parVar(){
$_pattern = '/\{\$([\w]+)\}/';
if (preg_match($_pattern,$this->_tpl)){
$this->_tpl = preg_replace($_pattern,"<?php echo \$this->_vars['$1'];?>",$this->_tpl);
}
}
//解析foreach语句
private function parForeach(){
$_patternForeach = '/\{foreach\s+\$([\w]+)\(([\w]+),([\w]+)\)\}/';
$_patternEndForeach = '/\{\/foreach\}/';
$_patternVar = '/\{@([\w]+)\}/';
if (preg_match($_patternForeach,$this->_tpl)){
if (!preg_match($_patternEndForeach,$this->_tpl)){
exit('ERROR:foreach语句没有闭合');
}
$this->_tpl = preg_replace($_patternForeach,'<?php foreach(\$this->_vars["$1"] as \$$2=>\$$3){?>',$this->_tpl);
$this->_tpl = preg_replace($_patternEndForeach,'<?php }?>',$this->_tpl);
//判断foreach是否有数据
if (preg_match($_patternVar,$this->_tpl)){
$this->_tpl = preg_replace($_patternVar,'<?php echo \$$1?>',$this->_tpl);
}
}
}
//解析include语句
private function parInclude(){
$_patternInclude = '/\{include\s+file=\"([\w\.\-]+)\"\}/';
if (preg_match($_patternInclude,$this->_tpl,$_file)){
if ((!file_exists($_file[1])) || empty($_file)){
exit('ERROR:文件'.$_file[1].'不存在');
}
$this->_tpl = preg_replace($_patternInclude,"<?php include '$1'?>",$this->_tpl);
}
}
//解析系统变量
private function parConfig(){
$_pattern = '/<!-- \{([\w]+)\} -->/';
if (preg_match($_pattern,$this->_tpl)){
$this->_tpl = preg_replace($_pattern,"<?php echo \$this->_config['$1'];?>",$this->_tpl);
}
}
}
1.引入模板解析类,并且实例化模板解析类,再将模板文件传入给模板解析类作为初始化参数,构造函数获取到模板文件类型,赋值给模板解析类的私有属性$_tpl
2.当在Template中调用Parser模板解析类中的compile()方法的时候,将编译文件名传递进去。
3.将模板文件放入编译文件之前,先解析tpl中的所有模板标签
public function compile($_parFile)
{
$this->parConfig();
$this->parInclude();
$this->parForeach();
$this->parCommon();
$this->parIf();
$this->parVar();
//生成编译文件之前解析模板内容
if (!file_put_contents($_parFile,$this->_tpl)){
exit('ERROR:编译文件生成出错');
}
}
已解析if语句为例:
{if $a}
<div>我是1号界面</div>
{else}
<div>我是2号界面</div>
{/if}
//解析if语句
private function parIf(){
$_pattenIf = '/\{if\s+\$([\w]+)\}/';
$_pattenEndIf = '/\{\/if\}/';
$_pattenElse = '/\{else\}/';
if (preg_match($_pattenIf,$this->_tpl)){
if (preg_match($_pattenEndIf,$this->_tpl)){
//匹配if开头
$this->_tpl = preg_replace($_pattenIf,'<?php if($this->_vars[\'$1\']){?>',$this->_tpl);
//匹配if结尾符并且更改为php形式
$this->_tpl = preg_replace($_pattenEndIf,"<?php };?>",$this->_tpl);
//匹配else
if (preg_match($_pattenElse,$this->_tpl)){
$this->_tpl = preg_replace($_pattenElse,"<?php }else{;?>",$this->_tpl);
}
}else{
exit('ERROR:IF语句没有关闭');
}
}
}
流程是先查找模板中是否存在{if $X}形式的模板标签,如果存在就查找{/if}
如果{/if}存在的情况在区分是否有else,然后通过字符串替换,将其变成PHP形式的输出
知识点:
preg_match() 函数用于进行正则表达式匹配,成功返回 1 ,否则返回 0 。
int preg_match( string pattern, string subject [, array matches ] )
pattern |
正则表达式 |
subject |
需要匹配检索的对象 |
matches |
可选,存储匹配结果的数组, $matches[0] 将包含与整个模式匹配的文本,$matches[1] 将包含与第一个捕获的括号中的子模式所匹配的文本,以此类推 |
preg_replace (正则表达式, 替换成, 字符串, 最大替换次数【默认-1,无数次】, 替换次数)
PHP模板解析入门的更多相关文章
- 【Ecmall】ECMall2.x模板制作入门系列(认识ECMall模板)
ECMall2.x模板制作入门系列之1(认识ECMall模板) 从ECMall2.0全新架构发布以来,随着版本的不断更新,ECMall已经逐渐走向一个稳定时期,是时候整理一些实用教程了.下面给大家带来 ...
- ECMall2.x模板制作入门系列之2(模板标签/语法)
ECMall2.x模板制作入门系列之2(模板标签/语法) 今天给大家带来一个模板语法的教程.希望能为ECMall模板制作者提供一份参考资料.如有问题.建议和意见,欢迎提出. 在ECMall模板中,用& ...
- OpenFaaS实战之七:java11模板解析
欢迎访问我的GitHub https://github.com/zq2599/blog_demos 内容:所有原创文章分类汇总及配套源码,涉及Java.Docker.Kubernetes.DevOPS ...
- vert.x学习(四),使用模板解析器ClassLoaderTemplateResolver
在vert.x中使用模板解析,可以为我们带来很多方便.我这里学习了一下ClassLoaderTemplateResolver的简单使用.这次工程配置与上篇一样,不需要做任何多的配置.直接编写代码就可以 ...
- smarty 模板的入门使用
<?php require_once 'inc/libs/Smarty.class.php'; $s=new Smarty(); // echo $s::SMARTY_VERSION; // e ...
- js的简单模板解析
在编程中总是会遇见很多动态生成的东西,一般我们都是通过简单的html拼接起来的 function createHtml(name, phone, addr, email, imageSrc){ var ...
- CI 模板解析器类
模板解析器类可以解析你的视图文件中的伪变量.它可以解析简单的变量或者以变量作为标签的结构.如果你以前没有用过模板引擎,那么伪变量如下所示: <html><head><ti ...
- PHP模板解析类实例
作者:mckee 这篇文章主要介绍了PHP模板解析类,涉及php针对模板文件的解析与字符串处理的相关技巧,具有一定参考借鉴价值,需要的朋友可以参考下 <?php class template { ...
- thymeleaf模板引擎入门
ThymeLeaf是什么 Thymeleaf是一个用于服务器端的java模板引擎,它使用简单但功能强大,目前可以处理的模板类型包括:HTML.XML.TEXT.JavaScript.CSS等. 搭建t ...
随机推荐
- [洛谷P4389]付公主的背包
题目大意:有$n(n\leqslant10^5)$种物品,第$i$个物品体积为$v_i$,都有$10^5$件.给定$m(m\leqslant10^5)$,对于$s\in [1,m]$,请你回答用这些商 ...
- [洛谷P4725]【模板】多项式对数函数
题目大意:给出$n-1$次多项式$A(x)$,求一个 $\bmod{x^n}$下的多项式$B(x)$,满足$B(x) \equiv \ln A(x)$.在$\bmod{998244353}$下进行.保 ...
- Educational Codeforces Round 2 A. Extract Numbers
打开题目链接 题意:输入一个字符串,用,或:分隔输出字符串和整数(不含前导0和浮点数) ACcode: #include <iostream> #include <cstdio> ...
- poj 3678 Katu Puzzle 2-SAT 建图入门
Description Katu Puzzle is presented as a directed graph G(V, E) with each edge e(a, b) labeled by a ...
- 文件排版(codevs 1300)
题目描述 Description 写电子邮件是有趣的,但不幸的是经常写不好看,主要是因为所有的行不一样长,你的上司想要发排版精美的电子邮件,你的任务是为他编写一个电子邮件排版程序. 完成这个任务最简单 ...
- foj Problem 2107 Hua Rong Dao
Problem 2107 Hua Rong Dao Accept: 503 Submit: 1054Time Limit: 1000 mSec Memory Limit : 32768 K ...
- 今天写页面,将.net也使用了语义法规来判断。做法来源是.net ado.net的类型判断 Infertype。一样的原理
做了一个封装,使页面更容易维护一些.主要是两个方法,如下所示: private void BindModule(Action<Repeater, string> bindSingRpt) ...
- 10.OpenStack块存储服务
添加块存储服务 安装和配置控制器节点 创建数据库 mysql -uroot -ptoyo123 CREATE DATABASE cinder; GRANT ALL PRIVILEGES ON cind ...
- Nginx 兼容IE8
前言 前段时间由于业务需要,在服务器上新增一个服务专门接收各个门店的业务结算数据,接口文档指明需要使用https协议.这本不是什么问题,因为之前服务器已经有配置过https.但等到服务部署之后才发现这 ...
- MSSQL—列记录合并成一行
在项目开发中,有时会碰到将列记录合并为一行的情况,例如根据地区将人员姓名合并,或根据拼音首字母合并城市等,下面就以根据地区将人员姓名合并为例,详细讲一下合并的方法. 首先,先建一个表,并添加一些数据, ...