php制作简单模版引擎


Posted in PHP onApril 07, 2016

PHP模板引擎就是一个PHP类库,使用它可以使PHP代码和HTML代码进行分离,使代码的可读性和维护性得到显著提高。而且这样做的好处是,让美工专心设计HTML前台页面,程序员专心去写PHP业务逻辑。因此,模化引擎很适合公司的Web开发团队使用,使每个人都能发挥其特长

下面我们就来看看如何简单的来实现php的模板引擎

parser.class.php

<?php
 
/**
 * 模版解析类
 */
class Parser
{
  // 字段,接收模版文件内容
  private $_tpl;
   
  // 构造方法,获取模版文件内容
  public function __construct($_tplFile)
  {
    if (! $this->_tpl = file_get_contents($_tplFile)) {
      exit('ERROR:模版文件读取错误');
    }
  }
   
  // 解析普通变量
  private function parvar()
  {
    $_patten = '/<!--\s+\{\$([\w]+)\}\s+-->/';
    if (preg_match($_patten,$this->_tpl)) {
      $this->_tpl = preg_replace($_patten, "<?php echo \$this->_vars['$1'];?>",$this->_tpl);
    }
  }
 
  //解析IF语句
  private function parif(){
    $_pattenif = '/<!--\s+\{if\s+\$([\w]+)\}\s+-->/';
    $_pattenElse = '/<!--\s+\{else\}\s+-->/';
    $_pattenEndif = '/<!--\s+\{\/if\}\s+-->/';
    if (preg_match($_pattenif,$this->_tpl)) {
      if (preg_match($_pattenEndif,$this->_tpl)) {
        $this->_tpl = preg_replace($_pattenif,"<?php if (\$this->_vars['$1']){?>",$this->_tpl);
        $this->_tpl = preg_replace($_pattenEndif,"<?php } ?>",$this->_tpl);
        if (preg_match($_pattenElse,$this->_tpl)) {
          $this->_tpl = preg_replace($_pattenElse,"<?php }else{?>",$this->_tpl);
        }
      }else{
      echo 'ERROR:IF语句没有关闭!';
      }
    }
  }
 
  //PHP注释解析
 
  private function parCommon(){
    $_pattenCommon = '/<!--\s+\{#\}(.*)\{#\}\s+-->/';
    if (preg_match($_pattenCommon,$this->_tpl)) {
      $this->_tpl = preg_replace($_pattenCommon,"<?php /* $1 */ ?>",$this->_tpl);
    }
  }
   
  //解析foreach语句
  private function parForeach(){
    $_pattenForeach = '/<!--\s+\{foreach\s+\$([\w]+)\(([\w]+),([\w]+)\)\}\s+-->/';
    $_pattenForeachEnd = '/<!--\s+\{\/foreach\}\s+-->/';
    $_pattenForeachValue = '/<!--\s+\{@([\w]+)\}\s+-->/';
    if (preg_match($_pattenForeach,$this->_tpl)) {
      if (preg_match($_pattenForeachEnd,$this->_tpl)) {
        $this->_tpl = preg_replace($_pattenForeach, "<?php foreach (\$this->_vars['$1'] as \$$2=>\$$3) {?>", $this->_tpl);
        $this->_tpl = preg_replace($_pattenForeachEnd, "<?php }?>", $this->_tpl);
        if (preg_match($_pattenForeachValue, $this->_tpl)) {
          $this->_tpl = preg_replace($_pattenForeachValue,"<?php echo \$$1;?>",$this->_tpl);
        }
      }else{
      echo 'ERROR:Foreach语句没有关闭!';  
      }
    }
  }
 
  //解析include方法
  private function parInclude(){
    $_pattenInclude = '/<!--\s+\{include\s+file=\"([\w\.\-]+)\"\}\s+-->/';
    if (preg_match($_pattenInclude,$this->_tpl,$_file,$_file)) {
      if (!file_exists($_file[1])||empty($_file)) {
        echo 'ERROR:包含文件出错!';
      }
      $this->_tpl = preg_replace($_pattenInclude,"<?php include '$1';?>",$this->_tpl);
    }
  }
 
  //解析系统变量方法
  private function parConfig(){
    $_pattenConfig = '/<!--\s+\{([\w]+)\}\s+-->/';
    if (preg_match($_pattenConfig,$this->_tpl)) {
      $this->_tpl = preg_replace($_pattenConfig,"<?php echo \$this->_config['$1'];?>",$this->_tpl);
    }
  }
  // 对外公共方法
  public function compile($_path)
  {
    // 解析模版文件
    $this->parvar();
    $this->parif();
    $this->parForeach();
    $this->parInclude();
    $this->parCommon();
    $this->parConfig();
    // 生成编译文件
    if (! file_put_contents($_path, $this->_tpl)) {
      exit('ERROR:编译文件生成错误!');
    }
  }
}
?>

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) || !is_dir(CONFIG)) {
      echo 'ERROR:模版目录或编译目录,缓存目录不存在!自动创建!'."<br />";
      if (!is_dir(TPL_DIR)) {
        mkdir(TPL_DIR);
        echo '模版目录'.TPL_DIR.'建立'."<br />";
      }
      if (!is_dir(TPL_C_DIR)) {
        mkdir(TPL_C_DIR);
        echo '编译目录'.TPL_C_DIR.'建立'."<br />";
      }
      if (!is_dir(CACHE)) {
        mkdir(CACHE);
        echo '缓存目录'.CACHE.'建立'."<br />";
      }
      if (!is_dir(CONFIG)) {
        mkdir(CONFIG);
        echo '缓存目录'.CONFIG.'建立'."<br />";
      }
      exit();
    }
    //保存系统变量
    $_sxe = simplexml_load_file(CONFIG.'/config.xml');
    $_tagLib = $_sxe->xpath('/root/taglib');
    foreach ($_tagLib as $_tag) {
      $this->_config["$_tag->name"] = $_tag->value;
    }
  }
 
  //assign()方法,用于注入变量
  public function assign($_var,$_value){
    //$_var用于同步模版里的变量名
    //$_value表示值
    if (isset($_var)&&!empty($_var)) {
      $this->_vars[$_var] = $_value;
    }else{
      exit('ERROR:设置模版变量!');
    }
 
  }
 
  //display()方法
  public function display($_file)
  {
    $_tplFile = TPL_DIR . $_file;
    // 判断文件是否存在
    if (! file_exists($_tplFile)) {
      echo 'ERROR:模版文件不存在!自动创建Index.tpl模版文件!';
      file_put_contents($_tplFile,'Index');
      exit();
    }
 
    //生成编译文件
    $_path = TPL_C_DIR.md5($_file).'-'.$_file.'.php';
    //缓存文件
    $_cacheFile = CACHE.md5($_file).'-'.$_file.'.html';
    //当第二次运行相同文件,直接载入缓存文件
    if (IS_CACHE) {
      //判断缓存文件和编译文件都存在
      if (file_exists($_cacheFile)&&file_exists($_path)) {
        //判断模版文件是否修改过
        if (filemtime($_path)>=filemtime($_tplFile)&&filemtime($_cacheFile)>=filemtime($_path)) {
          include $_cacheFile;
          echo '<!--cache-->';
          return;
        }
      }
    }
    //当编译文件不存在或者文件发生改变则重新生成
    if (!file_exists($_path)||filemtime($_path)<filemtime($_tplFile)) {
      require ROOT_PATH.'/Class/parser.class.php';
      //构造方法是传入模版文件地址
      $_parser = new Parser($_tplFile);
      //传入编译文件地址
      $_parser->compile($_path);
    }
    //载入编译文件
    include $_path;
    if (IS_CACHE) {
      //获取缓冲区数据
      file_put_contents($_cacheFile,ob_get_contents());
      //清楚缓冲区
      ob_end_clean();
      //载入缓存文件
      include $_cacheFile;
    }
  }
}
?>

templates.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('CONFIG',ROOT_PATH.'/Config/');
 
//是否开启缓冲区
define('IS_CACHE',false);//false
 
//判断是否需要开启
IS_CACHE ? ob_start() : null;
 
//引入模版类
require ROOT_PATH.'/Class/Templates.class.php';
 
//实例化模版类
$_tpl=new Templates();
 
$_tpl->display('index.tpl');
?>

templates/index.tpl

<!DOCTYPE html>
<html lang="zn-cn">
<head>
  <meta charset="UTF-8">
  <meta name="viewport" content="width=device-width, initial-scale=1, maximum-scale=1, user-scalable=no">
  <meta name="description" content="">
  <meta name="keywords" content="">
  <meta name="author" content="">
  <meta name="author" content="">
  <title><!-- {WebName} --></title>
  <link rel="stylesheet" type="text/css" href="css/bootstrap.min.css">
  <link rel="stylesheet" type="text/css" href="css/style.css">
  </head>
<body>
 
<!-- {#}php注释{#} -->
<!-- {if $a} -->
123
<!-- {else} -->
321
<!-- {/if} -->
<br />
<!-- {foreach $array(key,value)} -->
  <!-- {@key} -->...<!-- {@value} --><br />
<!-- {/foreach} -->
系统变量<!-- {WebName} --><br />
普通变量<!-- {$name} --><br />

  <script src="/js/jquery-2.2.1.min.js" type="text/javascript"></script>
  <script src="/js/bootstrap.min.js" type="text/javascript"></script>
  <script type="text/javascript">
  </script>
</body>
</html>

config/config.xml

<?xml version="1.0" encoding="UTF-8"?>
<root>
  <taglib>
    <name>WebName</name>
    <value>XXX网站</value>
  </taglib>
</root>
PHP 相关文章推荐
php Smarty模板生成html文档的方法
Apr 12 PHP
php输入流php://input使用浅析
Sep 02 PHP
PHP队列用法实例
Nov 05 PHP
自己写的php中文截取函数mb_strlen和mb_substr
Feb 09 PHP
ThinkPHP控制器详解
Jul 27 PHP
JSON用法之将PHP数组转JS数组,JS如何接收PHP数组
Oct 08 PHP
php倒计时出现-0情况的解决方法
Jul 28 PHP
PHP简单判断手机设备的方法
Aug 23 PHP
PHP使用imagick扩展实现合并图像的方法
Apr 25 PHP
详解php协程知识点
Sep 21 PHP
PDO::lastInsertId讲解
Jan 29 PHP
PHP pthreads v3下worker和pool的使用方法示例
Feb 21 PHP
thinkphp框架下实现登录、注册、找回密码功能
Apr 06 #PHP
非常有用的9个PHP代码片段
Apr 06 #PHP
10个对初学者非常有用的PHP技巧
Apr 06 #PHP
yii2.0实现pathinfo的形式访问的配置方法
Apr 06 #PHP
PHP实现的DES加密解密实例代码
Apr 06 #PHP
php使用正则验证中文
Apr 06 #PHP
php HTML无刷新提交表单
Apr 05 #PHP
You might like
《PHP边学边教》(01.开篇――准备工作)
2006/12/13 PHP
php使用PDO方法详解
2014/12/27 PHP
PHP 实现重载
2021/03/09 PHP
使用jQuery操作Cookies的实现代码
2011/10/09 Javascript
解决JS浮点数运算出现Bug的方法
2013/03/12 Javascript
基于jquery实现的文字淡入淡出效果
2013/11/14 Javascript
将字符串中由空格隔开的每个单词首字母大写
2014/04/06 Javascript
jQuery弹出框代码封装DialogHelper
2015/01/30 Javascript
jQuery验证元素是否为空的两种常用方法
2015/03/17 Javascript
Javascript中的迭代、归并方法详解
2016/06/14 Javascript
jQuery 选择符详细介绍及整理
2016/12/02 Javascript
浅谈Node.js:fs文件系统模块
2016/12/08 Javascript
Angular-Touch库用法示例
2016/12/22 Javascript
详解js正则表达式验证时间格式xxxx-xx-xx形式
2018/02/09 Javascript
vue生命周期实例小结
2018/08/15 Javascript
vue组件(全局,局部,动态加载组件)
2018/09/02 Javascript
react中使用css的7中方式(最全总结)
2019/02/11 Javascript
[05:46]2018完美盛典-《同梦共竞》
2018/12/17 DOTA
Python多进程分块读取超大文件的方法
2016/04/13 Python
asyncio 的 coroutine对象 与 Future对象使用指南
2016/09/11 Python
TensorFlow车牌识别完整版代码(含车牌数据集)
2019/08/05 Python
Django Haystack 全文检索与关键词高亮的实现
2020/02/17 Python
Django xadmin安装及使用详解
2020/10/26 Python
HTML5 placeholder(空白提示)属性介绍
2013/08/07 HTML / CSS
HTML5和以前HTML4的区别整理
2013/10/20 HTML / CSS
Java中有几种方法可以实现一个线程?用什么关键字修饰同步方法?stop()和suspend()方法为何不推荐使用?
2015/08/04 面试题
医学院学生的自我评价分享
2013/11/19 职场文书
英文自荐信
2013/12/19 职场文书
高级编程求职信模板
2014/02/16 职场文书
2014年学生管理工作总结
2014/12/20 职场文书
任命书怎么写
2015/03/02 职场文书
廉洁自律证明
2015/06/24 职场文书
教师读书笔记
2015/06/29 职场文书
新闻通讯稿范文
2015/07/22 职场文书
2016年10月份红领巾广播稿
2015/12/21 职场文书
vue点击弹窗自动触发点击事件的解决办法(模拟场景)
2021/05/25 Vue.js