文章詳細介紹了php合并靜態檔案所需要的相關配置以及完整代碼,以及使用說明,非常的詳盡,這裡推薦給有需要的小夥伴
配置PHP.ini
更改配置項(必須)auto_prepend_file = "C:\xampp\htdocs\auto_prepend_file.php"
更改配置項(可選)allow_url_include = On
auto_prepend_file.php檔案內容
代碼如下:
<?php/** * 引入static檔案 * @param {array|string} 相對路徑 * @param {string} 當前執行指令碼所在的路徑FILE * */function import_static($files, $path=NULL){ // 更改當前指令碼的執行路徑 $old_dir = getcwd(); $tmp_dir = (isset($path)) ? dirname($path): dirname(FILE); chdir($tmp_dir); // 整理包含檔案 if (!is_array($files)) { $tmp = array(); $tmp[] = $files; $files = $tmp; } // 發送頭資訊 if (isset($files[0])) { if (stripos($files[0], '.js') !== false) { $header_str = 'Content-Type: text/javascript'; } elseif (stripos($files[0], '.css') !== false) { $header_str = 'Content-Type: text/css'; } if (!ob_get_contents()) { header($header_str); } } // 引入包含檔案 foreach($files as $key=>$value) { require_once($value); } // 改回當前指令碼的執行路徑 chdir($old_dir);}?>
使用方法
代碼如下:
"a.js"、"b.js"和"../c.js"是待合并的JS檔案,將其合并為base.js.php,則base.js.php中的代碼如下:<?php import_static(array( 'a.js', 'b.js', '../c.js', '../moduleB/all.js.php' // 也可引用.php檔案 ), FILE);?>
在HTML頁面中使用<script type="text/javascript" src="base.js.php"></script>即可引入。
產品上線前,使用批次檔進行處理,主要做兩方面的工作
1.將"*.js.php"輸出到"*.js"檔案,並刪除"*.js.php"。命令列:php *.js.php > *.js
2.將HTML頁面中對"*.js.php"的引用替換為"*.js"。preg_replace()
PS:import_static函數解決了PHP中include()處理相對路徑的問題。
下面這個PHP的代碼執行個體,功能是協助使用者重設密碼,requestResetPassword是接收使用者重設密碼的請求並且做了相應的檢查。為了更好的複用性,我將重設密碼的操作單獨分配到一個新的resetPassword的函數,更改完密碼的後再調用sendEmail向使用者發送一封通知訊息。
/** * 使用者請求重設密碼的接收器 */function requestResetPassword() { //檢查使用者是否存在 if( !checkUserExists( $_GET['userid'] ) ) { exit('抱歉,使用者不存在,請確認使用者帳號。'); } resetPassword( $_GET['userid'] ); //最後向使用者發送一封郵件 sendEmail( $_GET['userid'], '重設密碼成功', '新的密碼是xxxx' ); exit('新密碼已經發送到你的郵箱。');}/** * 協助使用者重設密碼 */function resetPassword( $userid ) { //檢查使用者是否存在 if( !checkUserExists( $userid ) ) { return false; } //進行重設使用者密碼的操作 //略... return true;}/** * 向使用者發送一封郵件 */function sendEmail( $userid, $title, $content ) { //檢查使用者是否存在 if( !checkUserExists( $userid ) ) { return false; } //發送郵件操作 //略... return true;}/** * 檢查某個使用者是否存在 */function checkUserExists( $userid ) { $user = getUserInfo( $userid ); return !empty( $user );}/** * 擷取某個使用者的資料 */function getUserInfo( $userid ) { //假設我有一個query的函數,它用來查詢資料庫並返回資料 $user = query( "SELECT * FROM `user` WHERE `uid`=" . intval( $userid ) ); return is_array( $user ) ? $user : array() ;}
現在問題是,這三個函數都同時使用checkUserExists這個函數來檢查使用者不存在,資料庫查詢了三次,這樣帶來了一些額外的開銷。
如果要去掉三者之間任意一個checkUserExists,看上去是可能的。但是如果之後有某些功能要調用resetPassword或者sendEmail,使用者不存在時,系統可能會發生錯誤。
還有一個解決方案是,將resetPassword的邏輯寫到requestResetPassword裡,再過一點,把sendEmail的邏輯也寫進去。這樣函數調用減少,資料庫查詢也變成一次了,效能得到了提高。但是重設密碼和發送郵件的功能將不能得到複用,並且違背了單一責任的原則,代碼複雜度也提高了。
不過,因為函數分離和複用性都很好,如果實際效能受到影響,可能考慮用緩衝的方法減少資料庫查詢,我改動了它們共用的checkUserExists函數:
/** * 檢查某個使用者是否存在 */function checkUserExists( $userid ) { //增加一個緩衝,用以記錄檢查使用者的結果 static $cache = array(); //檢查目前使用者是否已經檢查過一次 if( isset( $cache[ $userid ] ) ) { return $cache[ $userid ]; } $user = getUserInfo( $userid ); //把結果記錄到緩衝中 $cache[ $userid ] = !empty( $user ); return $cache[ $userid ];}
也可以用同樣的方法改動getUserInfo函數。
這裡可以看到,當代碼的複用性提高時,想提高效能是很簡單的,效能的瓶頸也很容易被發現和修改。
儘管這個例子對效能影響還不夠大,還有一些影響更大的,比如說遍曆,我可能為了複用而將遍曆封裝到一個函數中,並且多次使用它。這些開銷對我的項目根本沒有預想中那樣有太大的影響,或者說是微乎其微的。所以我更願意把時間花在如何提高代碼的複用性和維護性方面,而不是糾結於浪費多這一點效能。實際效能如果真的達不到要求,也可以權衡增加硬體設定。