您好,登錄后才能下訂單哦!
本篇內容主要講解“構建可配置PHP應用程序的方式有哪些”,感興趣的朋友不妨來看看。本文介紹的方法操作簡單快捷,實用性強。下面就讓小編來帶大家學習“構建可配置PHP應用程序的方式有哪些”吧!
使用INI文件進行配置
PHP內建了對配置文件的支持。這是通過php.ini文件這樣的初始化文件(INI)機制實現的,在php.ini文件中定義了數據庫連接超時或會話如何存儲等常量。如果愿意的話,可以在這個php.ini文件中為應用程序定制配置。為了說明,我將下列代碼行添加到php.ini文件中。
myapptempdir=foo
然后,我編寫了一個小PHP腳本來讀取這個配置項,如清單1所示。
清單1.ini1.php
functionget_template_directory()
{
$v=get_cfg_var("myapptempdir");
return($v==null)?"tempdir":$v;
}
echo(get_template_directory()."\n");
?>
當在命令行中運行這段代碼時,得到如下結果:
%phpini1.php
foo
%
太棒了。但為什么不能用標準的INI函數來獲取myapptempdir配置項的值呢?我研究了一下,發現在大多數情況下,定制配置項不能使用這些方法來獲取。然而,使用get_cfg_var函數卻是可以訪問的。
為使這個方法更加簡單,將對變量的訪問封裝在第二個函數中,該函數使用配置鍵名及一個缺省值作為參數,如下所示。
清單2.ini2.php
functionget_ini_value($n,$dv)
{
$c=get_cfg_var($n);
return($c==null)?$dv:$c;
}
functionget_template_directory()
{
returnget_ini_value("myapptempdir","tempdir");
}
這是對如何訪問INI文件的一個很好的概括,所以,如果要使用一個不同的機制或將這個INI文件存儲到其他位置,就不需要為更改大量的函數而大費周折。
我不推薦使用INI文件作為應用程序的配置,這有兩個理由。首先,雖然這樣做較容易讀取INI文件,但卻幾乎不可能安全地寫INI文件。所以這樣做只適合于只讀配置項。第二,php.ini文件在服務器的所有應用程序上共享,所以我認為特定于應用程序的配置項不應該寫在該文件中。
需要對INI文件了解什么呢?最重要的是如何重置include路徑來添加配置項,如下所示。
清單3.ini3.php
echo(ini_get("include_path")."\n");
ini_set("include_path",
ini_get("include_path").":./mylib");
echo(ini_get("include_path")."\n");
?>
在本例中,我將我的本地mylib目錄添加到了include路徑中,所以能夠從該目錄中requirePHP文件,而不需要將該路徑添加到require語句中。
PHP中的配置
通常對于在INI文件中存儲配置條目的一個替代辦法是使用一個簡單的PHP腳本來保持數據。如下是一個樣例。
清單4.config.php
#Specifythelocationofthetemporarydirectory
#
$TEMPLATE_DIRECTORY="tempdir";
?>
使用該常量的代碼如下所示。
清單5.php.php
require_once'config.php';
functionget_template_directory()
{
global$TEMPLATE_DIRECTORY;
return$TEMPLATE_DIRECTORY;
}
echo(get_template_directory()."\n");
?>
該代碼首先包含配置文件(config.php),接著就可以直接使用這些常量了。
使用這項技術有很多優勢。首先,如果某些人僅僅瀏覽config.php文件,該頁面是空白的。所以可以將config.php放到相同的文件中,并作為Web應用程序的根。第二,在任何編輯器中都可編輯,并且在一些編輯器中甚至具備語法著色及語法檢查功能。
這項技術的缺點是,這是一個像INI文件一樣的只讀技術。將數據從此文件中提取出來是輕而易舉的,但在該PHP文件中調整數據卻很困難,在一些情況下甚至是不可能的。
下面的替代方法顯示了如何編寫在本質上既可讀又可寫的配置系統。
文本文件
前面的兩個例子對于只讀配置條目都是合適的,但對于既讀又寫的配置參數來說又如何呢?首先,看看清單6中的文本配置文件。
清單6.config.txt
#Myapplication'sconfigurationfile
Title=MyApp
TemplateDirectory=tempdir
這是同INI文件相同的文件格式,但我自己編寫了輔助工具。為此,我創建了自己的Configuration類,如下所示。
清單7.text1.php
classConfiguration
{
private$configFile='config.txt';
private$items=array();
function__construct(){$this->parse();}
function__get($id){return$this->items[$id];}
functionparse()
{
$fh=fopen($this->configFile,'r');
while($l=fgets($fh))
{
if(preg_match('/^#/',$l)==false)
{
preg_match('/^(.*?)=(.*?)$/',$l,$found);
$this->items[$found[1]]=$found[2];
}
}
fclose($fh);
}
}
$c=newConfiguration();
echo($c->TemplateDirectory."\n");
?>
該代碼首先創建了一個Configuration對象。該構造函數接下來讀取config.txt并用解析過的文件內容來設置局部變量$items。
該腳本隨后尋找TemplateDirectory,這并沒有在對象中直接定義。因此,使用設置成'TemplateDirectory'的$id來調用神奇的__get方法,__get方法針對該鍵返回$items數組中的值。
這個__get方法特定于PHPV5環境,所以此腳本必須在PHPV5下運行。實際上,本文中所有的腳本都需要在PHPV5下運行。
當在命令行運行此腳本時,能看到下列結果:
%phptext1.php
tempdir
%
一切都在預料之中,該對象讀取config.txt文件,然后為TemplateDirectory配置項獲得正確的值。
但對于設置一個配置值,應該怎么做呢?在此類中建立一個新方法及一些新的測試代碼,就能夠得到這個功能,如下所示。
清單8.text2.php
classConfiguration
{
...
function__get($id){return$this->items[$id];}
function__set($id,$v){$this->items[$id]=$v;}
functionparse(){...}
}
$c=newConfiguration();
echo($c->TemplateDirectory."\n");
$c->TemplateDirectory='foobar';
echo($c->TemplateDirectory."\n");
?>
現在,有了一個__set函數,它是__get函數的“堂兄弟”。該函數并不為一個成員變量獲取值,當要設置一個成員變量時,才調用這個函數。底部的測試代碼設置值并打印出新值。
下面是在命令行中運行此代碼時出現的結果:
%phptext2.php
tempdir
foobar
%
太好了!但如何能將它存儲到文件中,從而將使這個改動固定下來呢?為此,需要寫文件并讀取它。用于寫文件的新函數,如下所示。
清單9.text3.php
classConfiguration
{
...
functionsave()
{
$nf='';
$fh=fopen($this->configFile,'r');
while($l=fgets($fh))
{
if(preg_match('/^#/',$l)==false)
{
preg_match('/^(.*?)=(.*?)$/',$l,$found);
$nf.=$found[1]."=".$this->items[$found[1]]."\n";
}
else
{
$nf.=$l;
}
}
fclose($fh);
copy($this->configFile,$this->configFile.'.bak');
$fh=fopen($this->configFile,'w');
fwrite($fh,$nf);
fclose($fh);
}
}
$c=newConfiguration();
echo($c->TemplateDirectory."\n");
$c->TemplateDirectory='foobar';
echo($c->TemplateDirectory."\n");
$c->save();
?>
新的save函數巧妙地操作config.txt。我并沒有僅用更新過的配置項重寫文件(這樣會移除掉注釋),而是讀取了這個文件并靈活地重寫了$items數組中的內容。這樣的話,就保留了文件中的注釋。
在命令行運行該腳本并輸出文本配置文件中的內容,能夠看到下列輸出。
清單10.保存函數輸出
%phptext3.php
tempdir
foobar
%catconfig.txt
#Myapplication'sconfigurationfile
Title=MyApp
TemplateDirectory=foobar
%
構建可配置PHP應用程序有哪些方式
原始的config.txt文件現在被新值更新了。
XML配置文件
盡管文本文件易于閱讀及編輯,但卻不如XML文件流行。另外,XML有眾多適用的編輯器,這些編輯器能夠理解標記、特殊符號轉義等等。所以配置文件的XML版本會是什么樣的呢?清單11顯示了XML格式的配置文件。
清單11.config.xml
tempdir
清單12顯示了使用XML來裝載配置設置的Configuration類的更新版。
清單12.xml1.php
classConfiguration
{
private$configFile='config.xml';
private$items=array();
function__construct(){$this->parse();}
function__get($id){return$this->items[$id];}
functionparse()
{
$doc=newDOMDocument();
$doc->load($this->configFile);
$cn=$doc->getElementsByTagName("config");
$nodes=$cn->item(0)->getElementsByTagName("*");
foreach($nodesas$node)
$this->items[$node->nodeName]=$node->nodeValue;
}
}
$c=newConfiguration();
echo($c->TemplateDirectory."\n");
?>
看起來XML還有另一個好處:代碼比文本版的代碼更為簡潔、容易。為保存這個XML,需要另一個版本的save函數,將結果保存為XML格式,而不是文本格式。
清單13.xml2.php
...
functionsave()
{
$doc=newDOMDocument();
$doc->formatOutput=true;
$r=$doc->createElement("config");
$doc->appendChild($r);
foreach($this->itemsas$k=>$v)
{
$kn=$doc->createElement($k);
$kn->appendChild($doc->createTextNode($v));
$r->appendChild($kn);
}
copy($this->configFile,$this->configFile.'.bak');
$doc->save($this->configFile);
}
...
這段代碼創建了一個新的XML文檔對象模型(DocumentObjectModel,DOM),然后將$items數組中的所有數據都保存到這個模型中。完成這些以后,使用save方法將XML保存為一個文件。
使用數據庫
最后的替代方式是使用一個數據庫保存配置元素的值。那首先要用一個簡單的模式來存儲配置數據。下面是一個簡單的模式。
清單14.schema.sql
DROPTABLEIFEXISTSsettings;
CREATETABLEsettings(
idMEDIUMINTNOTNULLAUTO_INCREMENT,
nameTEXT,
valueTEXT,
PRIMARYKEY(id)
);
這要求進行一些基于應用程序需求的調整。例如,如果想讓配置元素按照每個用戶進行存儲,就需要添加用戶ID作為額外的一列。
為了讀取及寫入數據,我編寫了如圖15所示的更新過的Configuration類。
清單15.db1.php
require_once('DB.php');
$dsn='mysql://root:password@localhost/config';
$db=&DB::Connect($dsn,array());
if(PEAR::isError($db)){die($db->getMessage());}
classConfiguration
{
private$configFile='config.xml';
private$items=array();
function__construct(){$this->parse();}
function__get($id){return$this->items[$id];}
function__set($id,$v)
{
global$db;
$this->items[$id]=$v;
$sth2=$db->prepare('DELETEFROMsettingsWHEREname=?');
$db->execute($sth2,$id);
if(PEAR::isError($db)){die($db->getMessage());}
$sth3=$db->prepare('INSERTINTOsettings(id,name,value)VALUES(0,?,?)');
$db->execute($sth3,array($id,$v));
if(PEAR::isError($db)){die($db->getMessage());}
}
functionparse()
{
global$db;
$doc=newDOMDocument();
$doc->load($this->configFile);
$cn=$doc->getElementsByTagName("config");
$nodes=$cn->item(0)->getElementsByTagName("*");
foreach($nodesas$node)
$this->items[$node->nodeName]=$node->nodeValue;
$res=$db->query('SELECTname,valueFROMsettings');
if(PEAR::isError($db)){die($db->getMessage());}
while($res->fetchInto($row)){
$this->items[$row[0]]=$row[1];
}
}
}
$c=newConfiguration();
echo($c->TemplateDirectory."\n");
$c->TemplateDirectory='newfoo';
echo($c->TemplateDirectory."\n");
?>
這實際上是一個混合的文本/數據庫解決方案。請仔細觀察parse方法。該類首先讀取文本文件來獲取初始值,然后讀取數據庫,進而將鍵更新為最新的值。在設置一個值后,鍵就從數據庫中移除掉,并添加一條具有更新過的值的新記錄。
觀察Configuration類如何通過本文的多個版本來發揮作用是一件有趣的事,該類能從文本文件、XML及數據庫中讀取數據,并一直保持相同的接口。我鼓勵您在開發中也使用具有相同穩定性的接口。對于對象的客戶機來說,這項工作具體是如何運行的是不明確的。關鍵的是對象與客戶機之間的契約。
到此,相信大家對“構建可配置PHP應用程序的方式有哪些”有了更深的了解,不妨來實際操作一番吧!這里是億速云網站,更多相關內容可以進入相關頻道進行查詢,關注我們,繼續學習!
免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。