本文实例讲述了PHP实现简单的模板引擎功能。分享给大家供大家参考,具体如下:

php web开发中广泛采取mvc的设计模式,controller传递给view层的数据,必须通过模板引擎才能解析出来。实现一个简单的仅仅包含if,foreach标签,解析$foo变量的模板引擎。
编写template模板类和compiler编译类。代码如下:
<?php
namespace foo\base;
use foo\base\Object;
use foo\base\Compiler;
/**
*
*/
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 foo\base;
use foo\base\Object;
/**
*
*/
class Compiler extends Object
{
private $_content;
private $_valueMap = [];
private $_patten = [
'#\{\\$([a-zA-Z_\x7f-\xff][a-zA-Z0-9_\x7f-\xff]*)\}#',
'#\{if (.*?)\}#',
'#\{(else if|elseif) (.*?)\}#',
'#\{else\}#',
'#\{foreach \\$([a-zA-Z_\x7f-\xff][a-zA-Z0-9_\x7f-\xff]*)}#',
'#\{\/(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方法进行模板编译了。
/**
* [render 渲染模板文件]
* @param [type] $file [待编译的文件]
* @param [type] $values [键值对]
* @param array $templateConfig [编译配置]
* @return [type] [description]
*/
protected function render($file, $values, $templateConfig = [])
{
$di = Container::getInstance();
//依赖注入实例化对象
$di->template = function () use ($di, $templateConfig) {
$di->compiler = 'foo\base\Compiler';
$compiler = $di->compiler;
return new \foo\base\Template($compiler, $templateConfig);
};
$di->template->assign($values)->show($file);
}
Container类如下:
<?php
namespace foo\base;
use foo\base\Object;
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('test\index', ['name' => 'tom', 'age' => 20, 'friends' => ['jack', 'rose']], ['cacheTime' => 10]);
编写视图模板文件'test\index':
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<title>Document</title>
</head>
<body>
<p>展示模板文件视图</p>
<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基于pdo操作数据库技巧总结》、《PHP运算与运算符用法总结》、《PHP网络编程技巧总结》、《PHP基本语法入门教程》、《php面向对象程序设计入门教程》、《php字符串(string)用法总结》、《php+mysql数据库操作入门教程》及《php常见数据库操作技巧汇总》
希望本文所述对大家PHP程序设计有所帮助。
# PHP
# 模板引擎
# 需要使用php模板的朋友必看的很多个顶级PHP模板引擎比较分析
# PHP原生模板引擎 最简单的模板引擎
# PHP中MVC模式的模板引擎开发经验分享
# php模板引擎技术简单实现
# Pain 全世界最小最简单的PHP模板引擎 (普通版)
# CodeIgniter使用phpcms模板引擎
# php smarty模板引擎的6个小技巧
# 简单的自定义php模板引擎
# PHP的自定义模板引擎
# 自定义min版smarty模板引擎MinSmarty.class.php文件及用法
# 键值
# 递归
# 默认值
# 不存在
# 程序设计
# 就可以
# 是一个
# 好了
# 相关内容
# 感兴趣
# 要想
# 给大家
# 更多关于
# 所述
# 绑定
# 创建一个
# 若无
# 面向对象
# 就写
# 运算符
相关文章:
利用JavaScript实现拖拽改变元素大小
行程制作网站有哪些,第三方机票电子行程单怎么开?
c++ stringstream用法详解_c++字符串与数字转换利器
平台云上自主建站:模板化设计与智能工具打造高效网站
5种Android数据存储方式汇总
如何自定义建站之星网站的导航菜单样式?
如何在云指建站中生成FTP站点?
儿童网站界面设计图片,中国少年儿童教育网站-怎么去注册?
高防服务器如何保障网站安全无虞?
网站制作模板下载什么软件,ppt模板免费下载网站?
济南企业网站制作公司,济南社保单位网上缴费步骤?
如何彻底卸载建站之星软件?
建站上传速度慢?如何优化加速网站加载效率?
如何在服务器上配置二级域名建站?
Android自定义listview布局实现上拉加载下拉刷新功能
如何通过wdcp面板快速创建网站?
建站之星手机一键生成:多端自适应+小程序开发快速建站指南
网站制作公司哪里好做,成都网站制作公司哪家做得比较好,更正规?
网站按钮制作软件,如何实现网页中按钮的自动点击?
如何选择高效响应式自助建站源码系统?
C++如何使用std::optional?(处理可选值)
存储型VPS适合搭建中小型网站吗?
青浦网站制作公司有哪些,苹果官网发货地是哪里?
专业网站制作服务公司,有哪些网站可以免费发布招聘信息?
,想在网上投简历,哪几个网站比较好?
如何选择建站程序?包含哪些必备功能与类型?
武汉网站设计制作公司,武汉有哪些比较大的同城网站或论坛,就是里面都是武汉人的?
用v-html解决Vue.js渲染中html标签不被解析的问题
微信小程序 五星评分(包括半颗星评分)实例代码
javascript中的try catch异常捕获机制用法分析
攀枝花网站建设,攀枝花营业执照网上怎么年审?
临沂网站制作公司有哪些,临沂第四中学官网?
建站之星如何配置系统实现高效建站?
C#如何在一个XML文件中查找并替换文本内容
零服务器AI建站解决方案:快速部署与云端平台低成本实践
如何用西部建站助手快速创建专业网站?
制作ppt免费网站有哪些,有哪些比较好的ppt模板下载网站?
详解免费开源的DotNet二维码操作组件ThoughtWorks.QRCode(.NET组件介绍之四)
网站企业制作流程,用什么语言做企业网站比较好?
网站制作大概要多少钱一个,做一个平台网站大概多少钱?
如何基于PHP生成高效IDC网络公司建站源码?
购物网站制作费用多少,开办网上购物网站,需要办理哪些手续?
制作网站的过程怎么写,用凡科建站如何制作自己的网站?
成都网站制作公司哪家好,四川省职工服务网是做什么用?
股票网站制作软件,网上股票怎么开户?
深圳防火门网站制作公司,深圳中天明防火门怎么编码?
测试制作网站有哪些,测试性取向的权威测试或者网站?
如何通过西部建站助手安装IIS服务器?
青岛网站建设如何选择本地服务器?
c# 在ASP.NET Core中管理和取消后台任务
*请认真填写需求信息,我们会在24小时内与您取得联系。