PHP模板引擎如何实现


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

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

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

谢谢

模板引擎 php

wthall 10 years, 11 months ago

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

shonll answered 10 years, 11 months ago

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

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

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

神楽坂理子 answered 10 years, 11 months ago

举个简单的列子给你吧


 <?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);
}
?>

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

php 使用示例


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

模版使用示例


 
<html> <head> <title>{title}</title> <meta http-equiv="Content-Type" content="text/html;charset=utf-8"/> </head> <body> 你好,仅供参考<br/> {a.a} {a.b.c}<br/> {foreach a:$v} {autoindex}.{v}<br/> {/foreach} </body> </html> {include:footer}
阿斯特拉塔 answered 10 years, 11 months ago

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

c650187 answered 10 years, 11 months ago

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


 <!DOCTYPE HTML>
<title><?=$title?></title>

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

只要:


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

即可。

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


 ob_start();

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

$content = ob_get_clean();

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

real圣 answered 10 years, 11 months ago

摘自Slim:


 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();
}

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

丶Shana answered 10 years, 11 months ago

简单理解一下,是这样的


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

include 'file.html';
```



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

file.html就是file.php的模板

再换个思路是这样的


 <?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';


 <!-- file.html -->
{{var}}

fateben answered 10 years, 11 months ago

Your Answer