Home  >  Article  >  Backend Development  >  PHP模板引擎如何实现

PHP模板引擎如何实现

WBOY
WBOYOriginal
2016-06-06 20:38:501471browse

假设使用标签形式作为模板语法

在解析环节我想到两种实现方式
1. 利用正则获取标签内容,然后将标签进行规则性的替换并使用eval进行解析,解析完成后再用正则进行替换
2. 利用正则获取标签内容,然后将标签进行规则性的替换,替换完成后使用cli执行脚本并返回内容

现在的问题是,不太清楚整个流程和这两种实现方式是否正确,或者有别的方式。

谢谢

回复内容:

假设使用标签形式作为模板语法

在解析环节我想到两种实现方式
1. 利用正则获取标签内容,然后将标签进行规则性的替换并使用eval进行解析,解析完成后再用正则进行替换
2. 利用正则获取标签内容,然后将标签进行规则性的替换,替换完成后使用cli执行脚本并返回内容

现在的问题是,不太清楚整个流程和这两种实现方式是否正确,或者有别的方式。

谢谢

简单理解一下,是这样的

<code><?php // file.php
$var = 'text';

include 'file.html';
```



```
<!-- file.html -->
=$var?>
</code>

file.html就是file.php的模板

再换个思路是这样的

<code><?php // file.php

$var = 'text';

#complie template
if (!is_file('template_file.php')) {
    $contents = file_get_contents('file.html');
    $contents = preg_replace(.....);
    file_put_contents('template_file.php',$contents);
}

include 'template_file.php';
</code></code>
<code><!-- file.html -->
{{var}}
</code>

摘自Slim:

<code>    protected function render($template, $data = null)
{
    $templatePathname = $this->getTemplatePathname($template);
    if (!is_file($templatePathname)) {
        throw new \RuntimeException("View cannot render `$template` because the template does not exist");
    }

    $data = array_merge($this->data->all(), (array) $data);
    extract($data);
    ob_start();
    require $templatePathname;

    return ob_get_clean();
}
</code>

拿到数据之后要进一步处理还是display就看你自己了

举个简单的列子给你吧

<code><?php /****************
 * 核心文件
 * @discription: 编写简单的模板引擎
 */
define('INVIEW', true);
class view {
    var $tpl_dir = 'template';
    var $cache_dir = 'cache';
    var $tpl_ext = '.html';
    var $var_left = '{';
    var $var_right = '}';
    function __construct($config=array()) {
        extract($config);
        if(isset($tpl_dir))$this->tpl_dir = $tpl_dir;
        if(isset($cache_dir))$this->cache_dir = $cache_dir;
        if(isset($tpl_ext))$this->tpl_ext = $tpl_ext;
        if(isset($var_left))$this->var_left = $var_left;
        if(isset($var_right))$this->var_right = $var_right;
    }
    function load($tplfilename) {
        $tplfile = $this->tpl_dir.'/'.$tplfilename.$this->tpl_ext;
        if(!file_exists($tplfile))
            die('Template not found: '.$tplfile);
        return $this->cache($tplfilename, $tplfile);
    }
    //判断模板是否缓存,如模板文件有更改则重新编译
    function cache($tplname, $tpl_file) {
        $cache_file = $this->cache_dir.'/'.md5($tplname).'.php';
        if(!file_exists($cache_file) || filemtime($tpl_file)>filemtime($cache_file))
            $this->compile($tpl_file, $cache_file);
        return $cache_file;
    }
    //编译模板内容到PHP格式,并写入缓存
    function compile($tpl, $cache) {
        $body = file_get_contents($tpl);
        $vl = $this->var_left;
        $vr = $this->var_right;
        $patterns = array(
            "#$vl\s*include:(.+?)\s*$vr#i",
            "#$vl\s*if\s+(.+?)\s*$vr#i",
            "#$vl\s*else\s*$vr#i",
            "#$vl\s*elseif\s+(.+?)\s*$vr#i",
            "#$vl\s*endif\s*$vr#i",
            "#$vl\s*/if\s*$vr#i",
            "#$vl\s*foreach\s+(.+?):(.+?)\s*$vr#i",
            "#$vl\s*endforeach\s*$vr#i",
            "#$vl\s*/foreach\s*$vr#i",
            "#$vl([0-9a-zA-Z_]+?)\.([0-9a-zA-Z_]+?)\.([0-9a-zA-Z_]+?)$vr#i",
            "#$vl([0-9a-zA-Z_]+?)\.([0-9a-zA-Z_]+?)$vr#i",
            "#$vl([0-9a-zA-Z_\[\]\'\"]+?)$vr#i",
            "#$vl([0-9a-zA-Z_]+?):(.*?)$vr#i"
        );
        $replacements = array(
            "<?php include show('\\1'); ?>",
            "<?php if(\\1): ?>",
            "<?php else: ?>",
            "<?php elseif(\\1): ?>",
            "<?php endif; ?>",
            "<?php endif; ?>",
            "<?php if(count($\\1)>0):\$autoindex=0;foreach($\\1 as \\2):\$autoindex++; ?>",
            "<?php endforeach;endif; ?>",
            "<?php endforeach;endif; ?>",
            "<?php echo $\\1['\\2']['\\3']; ?>",
            "<?php echo $\\1['\\2']; ?>",
            "<?php echo $\\1; ?>",
            "<?php echo \\1(\\2); ?>"
        );
        $body = preg_replace($patterns, $replacements, $body);
        file_put_contents($cache, "<?php if(!defined('INVIEW'))die('YIQU'); ?>".$body);
    }
}
$view = new view();
function show($tpl) {
    global $view;
    return $view->load($tpl);
}
?>
</code>

上面就是一个模版引擎了,看看如何使用

php 使用示例

<code><br><?php include 'view.php';
$title = '文档标题';
$a=array('a'=>'rows','b'=>array('c'=>'inarray'));
include show('index');
?>

</code>

模版使用示例

<code><br>

<title>{title}</title>
<meta http-equiv="Content-Type" content="text/html;charset=utf-8">


你好,仅供参考<br>
{a.a}
{a.b.c}<br>
{foreach a:$v}
{autoindex}.{v}<br>
{/foreach}


{include:footer}

</code>

PHP 本身就可看作一個「模板引擎」。

<code>
<title>=$title?></title>

<main>=$main?></main>
</code>

只要:

<code>$title = 'test'; $main = '<input>';
require('template.inc.php');
</code>

即可。

如需要得到內容而非直接輸出,只要:

<code>ob_start();

$title = 'test'; $main = '<input>';
require('template.inc.php');

$content = ob_get_clean();
</code>

包裝成函數,從功能上看,算不算用 PHP 實現了模板引擎呢?

php模板一般是翻譯型,是把特定的文件翻譯成php或html,然後輸出的是那個文件的結果

php本身可以看作是C语言的模板。。。

楼主说的模板引擎,大抵都是用正则表达式实现的。

{$user_name}--->正则匹配--->
没什么太高深的东西在里面,如果你愿意,直接采用原生模式嵌入到html即可,毕竟正则效率也不是那么高

我很多年前写的c++版本,供参考,实现比较土,编译原理都还给老师了
https://github.com/pi1ot/webapplib/blob/master/waTemplate.h
https://github.com/pi1ot/webapplib/blob/master/waTemplate.cpp

Statement:
The content of this article is voluntarily contributed by netizens, and the copyright belongs to the original author. This site does not assume corresponding legal responsibility. If you find any content suspected of plagiarism or infringement, please contact admin@php.cn