search

Home  >  Q&A  >  body text

PHP模板引擎如何实现

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

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

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

谢谢

高洛峰高洛峰2897 days ago332

reply all(7)I'll reply

  • PHP中文网

    PHP中文网2017-04-10 15:19:38

    简单理解一下,是这样的

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

    reply
    0
  • 阿神

    阿神2017-04-10 15:19:38

    举个简单的列子给你吧

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

    reply
    0
  • 伊谢尔伦

    伊谢尔伦2017-04-10 15:19:38

    摘自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就看你自己了

    reply
    0
  • PHP中文网

    PHP中文网2017-04-10 15:19:38

    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 實現了模板引擎呢?

    reply
    0
  • ringa_lee

    ringa_lee2017-04-10 15:19:38

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

    reply
    0
  • 迷茫

    迷茫2017-04-10 15:19:38

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

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

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

    reply
    0
  • ringa_lee

    ringa_lee2017-04-10 15:19:38

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

    reply
    0
  • Cancelreply