0

0

php 编写一个简单的模板引擎

PHP中文网

PHP中文网

发布时间:2016-05-26 08:18:19

|

1236人浏览过

|

来源于php中文网

原创

php web开发中广泛采取mvc的设计模式,controller传递给view层的数据,必须通过模板引擎才能解析出来。实现一个简单的仅仅包含if,foreach标签,解析$foo变量的模板引擎。 
编写template模板类和compiler编译类。代码如下:

<?php

namespace fooase;
use fooaseObject;
use fooaseCompiler;
/**
* 
*/
class Template extends Object
{
    private $_config = [
        'suffix' => '.php',//文件后缀名
        'templateDir' => '../views/',//模板所在文件夹
        'compileDir' => '../runtime/cache/views/',//编译后存放的目录
        'suffixCompile' => '.php',//编译后文件后缀
        'isReCacheHtml' => false,//是否需要重新编译成静态html文件
        'isSupportPhp' => true,//是否支持php的语法
        'cacheTime' => 0,//缓存时间,单位秒
    ];
    private $_file;//带编译模板文件
    private $_valueMap = [];//键值对
    private $_compiler;//编译器

    public function __construct($compiler, $config = [])
    {
        $this->_compiler = $compiler;
        $this->_config = array_merge($this->_config, $config);
    }

    /**
     * [assign 存储控制器分配的键值]
     * @param  [type] $values [键值对集合]
     * @return [type]         [description]
     */
    public function assign($values)
    {
        if (is_array($values)) {
            $this->_valueMap = $values;
        } else {
            throw new Exception('控制器分配给视图的值必须为数组!');
        }
        return $this;
    }

    /**
     * [show 展现视图]
     * @param  [type] $file [带编译缓存的文件]
     * @return [type]       [description]
     */
    public function show($file)
    {
        $this->_file = $file;
        if (!is_file($this->path())) {
            throw new Exception('模板文件'. $file . '不存在!');
        }

        $compileFile = $this->_config['compileDir'] . md5($file) . $this->_config['suffixCompile'];
        $cacheFile = $this->_config['compileDir'] . md5($file) . '.html';

        //编译后文件不存在或者缓存时间已到期,重新编译,重新生成html静态缓存
        if (!is_file($compileFile) || $this->isRecompile($compileFile)) {
            $this->_compiler->compile($this->path(), $compileFile, $this->_valueMap);
            $this->_config['isReCacheHtml'] = true;

            if ($this->isSupportPhp()) {
                extract($this->_valueMap, EXTR_OVERWRITE);//从数组中将变量导入到当前的符号表
            }

        }

        if ($this->isReCacheHtml()) {
            ob_start();
            ob_clean();
            include($compileFile);
            file_put_contents($cacheFile, ob_get_contents());
            ob_end_flush();
        } else {
            readfile($cacheFile);
        }
    }

    /**
     * [isRecompile 根据缓存时间判断是否需要重新编译]
     * @param  [type]  $compileFile [编译后的文件]
     * @return boolean              [description]
     */
    private function isRecompile($compileFile)
    {
        return time() - filemtime($compileFile) > $this->_config['cacheTime'];
    }

    /**
     * [isReCacheHtml 是否需要重新缓存静态html文件]
     * @return boolean [description]
     */
    private function isReCacheHtml()
    {
        return $this->_config['isReCacheHtml'];
    }

    /**
     * [isSupportPhp 是否支持php语法]
     * @return boolean [description]
     */
    private function isSupportPhp()
    {
        return $this->_config['isSupportPhp'];
    }

    /**
     * [path 获得模板文件路径]
     * @return [type] [description]
     */
    private function path()
    {
        return $this->_config['templateDir'] . $this->_file . $this->_config['suffix'];
    }


}
<?php

namespace fooase;
use fooaseObject;

/**
* 
*/
class Compiler extends Object
{
    private $_content;
    private $_valueMap = [];
    private $_patten = [
        '#{\$([a-zA-Z_-][a-zA-Z0-9_-]*)}#',
        '#{if (.*?)}#',
        '#{(else if|elseif) (.*?)}#',
        '#{else}#',
        '#{foreach \$([a-zA-Z_-][a-zA-Z0-9_-]*)}#',
        '#{/(foreach|if)}#',
        '#{\^(k|v)}#',
    ];
    private $_translation = [
        "<?php echo $this->_valueMap['\1']; ?>",
        '<?php if (\1) {?>',
        '<?php } else if (\2) {?>',
        '<?php }else {?>',
        "<?php foreach ($this->_valueMap['\1'] as $k => $v) {?>",
        '<?php }?>',
        '<?php echo $\1?>'
    ];

    /**
     * [compile 编译模板文件]
     * @param  [type] $source   [模板文件]
     * @param  [type] $destFile [编译后文件]
     * @param  [type] $values   [键值对]
     * @return [type]           [description]
     */
    public function compile($source, $destFile, $values)
    {
        $this->_content = file_get_contents($source);
        $this->_valueMap = $values;

        if (strpos($this->_content, '{$') !== false) {
            $this->_content = preg_replace($this->_patten, $this->_translation, $this->_content);
        }
        file_put_contents($destFile, $this->_content);
    }
}

我们的控制器就可以调用template中的assign方法进行赋值,show方法进行模板编译了。

<?php
namespace fooase;
use fooaseObject;

class Container extends Object
{
    private static $_instance;
    private $s = [];
    public static $instances = [];

    public static function getInstance()
    {
        if (!(self::$_instance instanceof self)) {
            self::$_instance = new self();
        }
        return self::$_instance;
    }

    private function __construct(){}

    private function __clone(){}

    public function __set($k, $c)
    {
        $this->s[$k] = $c;
    }

    public function __get($k)
    {
        return $this->build($this->s[$k]);
    }

    /**
     * 自动绑定(Autowiring)自动解析(Automatic Resolution)
     *
     * @param string $className
     * @return object
     * @throws Exception
     */
    public function build($className)
    {     
        // 如果是闭包函数(closures)
        if ($className instanceof Closure) {
            // 执行闭包函数
            return $className($this);
        }

        if (isset(self::$instances[$className])) {
            return self::$instances[$className];
        }

        /** @var ReflectionClass $reflector */
        $reflector = new ReflectionClass($className);

        // 检查类是否可实例化, 排除抽象类abstract和对象接口interface
        if (!$reflector->isInstantiable()) {
            throw new Exception($reflector . ': 不能实例化该类!');
        }

        /** @var ReflectionMethod $constructor 获取类的构造函数 */
        $constructor = $reflector->getConstructor();

        // 若无构造函数,直接实例化并返回
        if (is_null($constructor)) {
            return new $className;
        }

        // 取构造函数参数,通过 ReflectionParameter 数组返回参数列表
        $parameters = $constructor->getParameters();

        // 递归解析构造函数的参数
        $dependencies = $this->getDependencies($parameters);

        // 创建一个类的新实例,给出的参数将传递到类的构造函数。
        $obj = $reflector->newInstanceArgs($dependencies);
        self::$instances[$className] = $obj;

        return $obj;
    }

    /**
     * @param array $parameters
     * @return array
     * @throws Exception
     */
    public function getDependencies($parameters)
    {
        $dependencies = [];

        /** @var ReflectionParameter $parameter */
        foreach ($parameters as $parameter) {
            /** @var ReflectionClass $dependency */
            $dependency = $parameter->getClass();

            if (is_null($dependency)) {
                // 是变量,有默认值则设置默认值
                $dependencies[] = $this->resolveNonClass($parameter);
            } else {
                // 是一个类,递归解析
                $dependencies[] = $this->build($dependency->name);
            }
        }

        return $dependencies;
    }

    /**
     * @param ReflectionParameter $parameter
     * @return mixed
     * @throws Exception
     */
    public function resolveNonClass($parameter)
    {
        // 有默认值则返回默认值
        if ($parameter->isDefaultValueAvailable()) {
            return $parameter->getDefaultValue();
        }

        throw new Exception('I have no idea what to do here.');
    }
}

要想以键值对的方式访问对象的属性必须实现ArrayAccess接口的四个方法, 
Object基类代码如下:

public function offsetExists($offset) 
    {
        return array_key_exists($offset, get_object_vars($this));
    }

    public function offsetUnset($key) 
    {
        if (array_key_exists($key, get_object_vars($this)) ) {
            unset($this->{$key});
        }
    }

    public function offsetSet($offset, $value) 
    {
        $this->{$offset} = $value;
    }

    public function offsetGet($var) 
    {
        return $this->$var;
    }

在某一控制器中就可以调用父类Controller的render方法啦

$this->render('testindex', ['name' => 'tom', 'age' => 20, 'friends' => ['jack', 'rose']], ['cacheTime' => 10]);

编写视图模板文件‘testindex’:

立即学习PHP免费学习笔记(深入)”;

<!DOCTYPE html>
<html>
<head>
    <meta charset="UTF-8">
    <title>Document</title>
</head>
<body>
   <p>展示模板文件视图</p><div class="aritcle_card flexRow">
                                                        <div class="artcardd flexRow">
                                                                <a class="aritcle_card_img" href="/xiazai/code/11026" title="UQ云商B2B2C系统"><img
                                                                                src="https://img.php.cn/upload/webcode/000/000/015/176468760758998.png" alt="UQ云商B2B2C系统"  onerror="this.onerror='';this.src='/static/lhimages/moren/morentu.png'" ></a>
                                                                <div class="aritcle_card_info flexColumn">
                                                                        <a href="/xiazai/code/11026" title="UQ云商B2B2C系统">UQ云商B2B2C系统</a>
                                                                        <p>UQCMS云商是一款B2B2C电子商务软件 ,非常适合初创的创业者,个人及中小型企业。程序采用PHP+MYSQL,模板采用smarty模板,二次开发,简单方便,无需学习其他框架就可以自行模板设计。永久免费使用,操作简单,安全稳定。支持PC+WAP+微信三种浏览方式,支持微信公众号。</p>
                                                                </div>
                                                                <a href="/xiazai/code/11026" title="UQ云商B2B2C系统" class="aritcle_card_btn flexRow flexcenter"><b></b><span>下载</span> </a>
                                                        </div>
                                                </div> 
   <p>{$name}</p>
   <p>{$age}</p>
   <?php echo ++$age;?>
   {if $age > 18}
        <p>已成年</p>
    {else if $age < 10}
        <p>小毛孩</p>
   {/if}

   {foreach $friends} 
      <p>{^v} </p>
   {/foreach}
</body>
</html>

至此,一个简单的模板编译引擎就写好了。

PHP速学教程(入门到精通)
PHP速学教程(入门到精通)

PHP怎么学习?PHP怎么入门?PHP在哪学?PHP怎么学才快?不用担心,这里为大家提供了PHP速学教程(入门到精通),有需要的小伙伴保存下载就能学习啦!

下载

相关标签:

php

本站声明:本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
pixiv网页版官网登录与阅读指南_pixiv官网直达入口与在线访问方法
pixiv网页版官网登录与阅读指南_pixiv官网直达入口与在线访问方法

本专题系统整理pixiv网页版官网入口及登录访问方式,涵盖官网登录页面直达路径、在线阅读入口及快速进入方法说明,帮助用户高效找到pixiv官方网站,实现便捷、安全的网页端浏览与账号登录体验。

463

2026.02.13

微博网页版主页入口与登录指南_官方网页端快速访问方法
微博网页版主页入口与登录指南_官方网页端快速访问方法

本专题系统整理微博网页版官方入口及网页端登录方式,涵盖首页直达地址、账号登录流程与常见访问问题说明,帮助用户快速找到微博官网主页,实现便捷、安全的网页端登录与内容浏览体验。

135

2026.02.13

Flutter跨平台开发与状态管理实战
Flutter跨平台开发与状态管理实战

本专题围绕Flutter框架展开,系统讲解跨平台UI构建原理与状态管理方案。内容涵盖Widget生命周期、路由管理、Provider与Bloc状态管理模式、网络请求封装及性能优化技巧。通过实战项目演示,帮助开发者构建流畅、可维护的跨平台移动应用。

64

2026.02.13

TypeScript工程化开发与Vite构建优化实践
TypeScript工程化开发与Vite构建优化实践

本专题面向前端开发者,深入讲解 TypeScript 类型系统与大型项目结构设计方法,并结合 Vite 构建工具优化前端工程化流程。内容包括模块化设计、类型声明管理、代码分割、热更新原理以及构建性能调优。通过完整项目示例,帮助开发者提升代码可维护性与开发效率。

20

2026.02.13

Redis高可用架构与分布式缓存实战
Redis高可用架构与分布式缓存实战

本专题围绕 Redis 在高并发系统中的应用展开,系统讲解主从复制、哨兵机制、Cluster 集群模式及数据分片原理。内容涵盖缓存穿透与雪崩解决方案、分布式锁实现、热点数据优化及持久化策略。通过真实业务场景演示,帮助开发者构建高可用、可扩展的分布式缓存系统。

26

2026.02.13

c语言 数据类型
c语言 数据类型

本专题整合了c语言数据类型相关内容,阅读专题下面的文章了解更多详细内容。

29

2026.02.12

雨课堂网页版登录入口与使用指南_官方在线教学平台访问方法
雨课堂网页版登录入口与使用指南_官方在线教学平台访问方法

本专题系统整理雨课堂网页版官方入口及在线登录方式,涵盖账号登录流程、官方直连入口及平台访问方法说明,帮助师生用户快速进入雨课堂在线教学平台,实现便捷、高效的课程学习与教学管理体验。

14

2026.02.12

豆包AI网页版入口与智能创作指南_官方在线写作与图片生成使用方法
豆包AI网页版入口与智能创作指南_官方在线写作与图片生成使用方法

本专题汇总豆包AI官方网页版入口及在线使用方式,涵盖智能写作工具、图片生成体验入口和官网登录方法,帮助用户快速直达豆包AI平台,高效完成文本创作与AI生图任务,实现便捷智能创作体验。

524

2026.02.12

PostgreSQL性能优化与索引调优实战
PostgreSQL性能优化与索引调优实战

本专题面向后端开发与数据库工程师,深入讲解 PostgreSQL 查询优化原理与索引机制。内容包括执行计划分析、常见索引类型对比、慢查询优化策略、事务隔离级别以及高并发场景下的性能调优技巧。通过实战案例解析,帮助开发者提升数据库响应速度与系统稳定性。

53

2026.02.12

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
PHP课程
PHP课程

共137课时 | 12万人学习

JavaScript ES5基础线上课程教学
JavaScript ES5基础线上课程教学

共6课时 | 11.2万人学习

PHP新手语法线上课程教学
PHP新手语法线上课程教学

共13课时 | 0.9万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号