学习:李炎恢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模板解析入门的更多相关文章

  1. 【Ecmall】ECMall2.x模板制作入门系列(认识ECMall模板)

    ECMall2.x模板制作入门系列之1(认识ECMall模板) 从ECMall2.0全新架构发布以来,随着版本的不断更新,ECMall已经逐渐走向一个稳定时期,是时候整理一些实用教程了.下面给大家带来 ...

  2. ECMall2.x模板制作入门系列之2(模板标签/语法)

    ECMall2.x模板制作入门系列之2(模板标签/语法) 今天给大家带来一个模板语法的教程.希望能为ECMall模板制作者提供一份参考资料.如有问题.建议和意见,欢迎提出. 在ECMall模板中,用& ...

  3. OpenFaaS实战之七:java11模板解析

    欢迎访问我的GitHub https://github.com/zq2599/blog_demos 内容:所有原创文章分类汇总及配套源码,涉及Java.Docker.Kubernetes.DevOPS ...

  4. vert.x学习(四),使用模板解析器ClassLoaderTemplateResolver

    在vert.x中使用模板解析,可以为我们带来很多方便.我这里学习了一下ClassLoaderTemplateResolver的简单使用.这次工程配置与上篇一样,不需要做任何多的配置.直接编写代码就可以 ...

  5. smarty 模板的入门使用

    <?php require_once 'inc/libs/Smarty.class.php'; $s=new Smarty(); // echo $s::SMARTY_VERSION; // e ...

  6. js的简单模板解析

    在编程中总是会遇见很多动态生成的东西,一般我们都是通过简单的html拼接起来的 function createHtml(name, phone, addr, email, imageSrc){ var ...

  7. CI 模板解析器类

    模板解析器类可以解析你的视图文件中的伪变量.它可以解析简单的变量或者以变量作为标签的结构.如果你以前没有用过模板引擎,那么伪变量如下所示: <html><head><ti ...

  8. PHP模板解析类实例

    作者:mckee 这篇文章主要介绍了PHP模板解析类,涉及php针对模板文件的解析与字符串处理的相关技巧,具有一定参考借鉴价值,需要的朋友可以参考下 <?php class template { ...

  9. thymeleaf模板引擎入门

    ThymeLeaf是什么 Thymeleaf是一个用于服务器端的java模板引擎,它使用简单但功能强大,目前可以处理的模板类型包括:HTML.XML.TEXT.JavaScript.CSS等. 搭建t ...

随机推荐

  1. [洛谷P4389]付公主的背包

    题目大意:有$n(n\leqslant10^5)$种物品,第$i$个物品体积为$v_i$,都有$10^5$件.给定$m(m\leqslant10^5)$,对于$s\in [1,m]$,请你回答用这些商 ...

  2. [洛谷P4725]【模板】多项式对数函数

    题目大意:给出$n-1$次多项式$A(x)$,求一个 $\bmod{x^n}$下的多项式$B(x)$,满足$B(x) \equiv \ln A(x)$.在$\bmod{998244353}$下进行.保 ...

  3. Educational Codeforces Round 2 A. Extract Numbers

    打开题目链接 题意:输入一个字符串,用,或:分隔输出字符串和整数(不含前导0和浮点数) ACcode: #include <iostream> #include <cstdio> ...

  4. 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 ...

  5. 文件排版(codevs 1300)

    题目描述 Description 写电子邮件是有趣的,但不幸的是经常写不好看,主要是因为所有的行不一样长,你的上司想要发排版精美的电子邮件,你的任务是为他编写一个电子邮件排版程序. 完成这个任务最简单 ...

  6. foj Problem 2107 Hua Rong Dao

    Problem 2107 Hua Rong Dao Accept: 503    Submit: 1054Time Limit: 1000 mSec    Memory Limit : 32768 K ...

  7. 今天写页面,将.net也使用了语义法规来判断。做法来源是.net ado.net的类型判断 Infertype。一样的原理

    做了一个封装,使页面更容易维护一些.主要是两个方法,如下所示: private void BindModule(Action<Repeater, string> bindSingRpt)  ...

  8. 10.OpenStack块存储服务

    添加块存储服务 安装和配置控制器节点 创建数据库 mysql -uroot -ptoyo123 CREATE DATABASE cinder; GRANT ALL PRIVILEGES ON cind ...

  9. Nginx 兼容IE8

    前言 前段时间由于业务需要,在服务器上新增一个服务专门接收各个门店的业务结算数据,接口文档指明需要使用https协议.这本不是什么问题,因为之前服务器已经有配置过https.但等到服务部署之后才发现这 ...

  10. MSSQL—列记录合并成一行

    在项目开发中,有时会碰到将列记录合并为一行的情况,例如根据地区将人员姓名合并,或根据拼音首字母合并城市等,下面就以根据地区将人员姓名合并为例,详细讲一下合并的方法. 首先,先建一个表,并添加一些数据, ...