PHP實(shí)現(xiàn)簡單的模板引擎功能示例
本文實(shí)例講述了PHP實(shí)現(xiàn)簡單的模板引擎功能。分享給大家供大家參考,具體如下:
php web開發(fā)中廣泛采取mvc的設(shè)計(jì)模式,controller傳遞給view層的數(shù)據(jù),必須通過模板引擎才能解析出來。實(shí)現(xiàn)一個簡單的僅僅包含if,foreach標(biāo)簽,解析$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,//是否需要重新編譯成靜態(tài)html文件
'isSupportPhp' => true,//是否支持php的語法
'cacheTime' => 0,//緩存時(shí)間,單位秒
];
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('控制器分配給視圖的值必須為數(shù)組!');
}
return $this;
}
/**
* [show 展現(xiàn)視圖]
* @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';
//編譯后文件不存在或者緩存時(shí)間已到期,重新編譯,重新生成html靜態(tài)緩存
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);//從數(shù)組中將變量導(dǎo)入到當(dāng)前的符號表
}
}
if ($this->isReCacheHtml()) {
ob_start();
ob_clean();
include($compileFile);
file_put_contents($cacheFile, ob_get_contents());
ob_end_flush();
} else {
readfile($cacheFile);
}
}
/**
* [isRecompile 根據(jù)緩存時(shí)間判斷是否需要重新編譯]
* @param [type] $compileFile [編譯后的文件]
* @return boolean [description]
*/
private function isRecompile($compileFile)
{
return time() - filemtime($compileFile) > $this->_config['cacheTime'];
}
/**
* [isReCacheHtml 是否需要重新緩存靜態(tài)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);
}
}
我們的控制器就可以調(diào)用template中的assign方法進(jìn)行賦值,show方法進(jìn)行模板編譯了。
/**
* [render 渲染模板文件]
* @param [type] $file [待編譯的文件]
* @param [type] $values [鍵值對]
* @param array $templateConfig [編譯配置]
* @return [type] [description]
*/
protected function render($file, $values, $templateConfig = [])
{
$di = Container::getInstance();
//依賴注入實(shí)例化對象
$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)
{
// 如果是閉包函數(shù)(closures)
if ($className instanceof \Closure) {
// 執(zhí)行閉包函數(shù)
return $className($this);
}
if (isset(self::$instances[$className])) {
return self::$instances[$className];
}
/** @var ReflectionClass $reflector */
$reflector = new \ReflectionClass($className);
// 檢查類是否可實(shí)例化, 排除抽象類abstract和對象接口interface
if (!$reflector->isInstantiable()) {
throw new \Exception($reflector . ': 不能實(shí)例化該類!');
}
/** @var ReflectionMethod $constructor 獲取類的構(gòu)造函數(shù) */
$constructor = $reflector->getConstructor();
// 若無構(gòu)造函數(shù),直接實(shí)例化并返回
if (is_null($constructor)) {
return new $className;
}
// 取構(gòu)造函數(shù)參數(shù),通過 ReflectionParameter 數(shù)組返回參數(shù)列表
$parameters = $constructor->getParameters();
// 遞歸解析構(gòu)造函數(shù)的參數(shù)
$dependencies = $this->getDependencies($parameters);
// 創(chuàng)建一個類的新實(shí)例,給出的參數(shù)將傳遞到類的構(gòu)造函數(shù)。
$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)) {
// 是變量,有默認(rèn)值則設(shè)置默認(rèn)值
$dependencies[] = $this->resolveNonClass($parameter);
} else {
// 是一個類,遞歸解析
$dependencies[] = $this->build($dependency->name);
}
}
return $dependencies;
}
/**
* @param ReflectionParameter $parameter
* @return mixed
* @throws Exception
*/
public function resolveNonClass($parameter)
{
// 有默認(rèn)值則返回默認(rèn)值
if ($parameter->isDefaultValueAvailable()) {
return $parameter->getDefaultValue();
}
throw new \Exception('I have no idea what to do here.');
}
}
要想以鍵值對的方式訪問對象的屬性必須實(shí)現(xiàn)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;
}
在某一控制器中就可以調(diào)用父類Controller的render方法啦
編寫視圖模板文件'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>
至此,一個簡單的模板編譯引擎就寫好了。
更多關(guān)于PHP相關(guān)內(nèi)容感興趣的讀者可查看本站專題:《PHP模板技術(shù)總結(jié)》、《PHP基于pdo操作數(shù)據(jù)庫技巧總結(jié)》、《PHP運(yùn)算與運(yùn)算符用法總結(jié)》、《PHP網(wǎng)絡(luò)編程技巧總結(jié)》、《PHP基本語法入門教程》、《php面向?qū)ο蟪绦蛟O(shè)計(jì)入門教程》、《php字符串(string)用法總結(jié)》、《php+mysql數(shù)據(jù)庫操作入門教程》及《php常見數(shù)據(jù)庫操作技巧匯總》
希望本文所述對大家PHP程序設(shè)計(jì)有所幫助。
相關(guān)文章
PHP定時(shí)執(zhí)行任務(wù)的3種方法詳解
PHP不支持多線程,有時(shí)候處理問題不是那么方便,今天談?wù)撘幌翽HP定時(shí)執(zhí)行的方法,感興趣的小伙伴們可以參考一下2015-12-12
【CLI】利用Curl下載文件實(shí)時(shí)進(jìn)度條顯示的實(shí)現(xiàn)
這篇文章主要給大家介紹了關(guān)于【CLI】利用Curl下載文件實(shí)時(shí)進(jìn)度條顯示的相關(guān)資料,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧。2018-03-03
重新封裝zend_soap實(shí)現(xiàn)http連接安全認(rèn)證的php代碼
重新封裝zend_soap實(shí)現(xiàn)http連接安全認(rèn)證,需要的朋友可以參考下。2011-01-01
thinkphp框架實(shí)現(xiàn)刪除和批量刪除
這篇文章主要為大家詳細(xì)介紹了thinkPHP框架實(shí)現(xiàn)刪除和批量刪除的相關(guān)資料,需要的朋友可以參考下2016-06-06

