9個必須知道的實用PHP函數和功能

即使使用 PHP 多年,也會偶然發現一些未曾瞭解的函數和功能。其中有些是非常有用的,但沒有得到充分利用。並不是所有人都會從頭到尾一頁一頁地閱讀手冊和函數參考!


1、任意參數數目的函數


你可能已經知道,PHP 允許定義可選參數的函數。但也有完全允許任意數目的函數參數的方法。以下是可選參數的例子:


以下為引用的內容:


// function with 2 optional arguments
function foo($arg1 = , $arg2 = ) {


 echo “arg1: $arg1
“;
 echo “arg2: $arg2
“;


}


foo(hello,world);
/* prints:
arg1: hello
arg2: world
*/


foo();
/* prints:
arg1:
arg2:
*/
 



現在讓我們看看如何建立能夠接受任何參數數目的函數。這一次需要使用 func_get_args() 函數:


以下為引用的內容:


// yes, the argument list can be empty
function foo() {


 // returns an array of all passed arguments
 $args = func_get_args();


 foreach ($args as $k => $v) {
  echo “arg”.($k+1).”: $v
“;
 }


}


foo();
/* prints nothing */


foo(hello);
/* prints
arg1: hello
*/


foo(hello, world, again);
/* prints
arg1: hello
arg2: world
arg3: again
*/
 



2、使用 Glob() 查找文件


許多 PHP 函數具有長描述性的名稱。然而可能會很難說出 glob() 函數能做的事情,除非你已經通過多次使用並熟悉瞭它。可以把它看作是比 scandir() 函數更強大的版本,可以按照某種模式搜索文件。


以下為引用的內容:


// get all php files
$files = glob(*.php);


print_r($files);
/* output looks like:
Array
(
    [0] => phptest.php
    [1] => pi.php
    [2] => post_output.php
    [3] => test.php
)
*/
 



你可以像這樣獲得多個文件:


以下為引用的內容:


// get all php files AND txt files
$files = glob(*.{php,txt}, GLOB_BRACE);


print_r($files);
/* output looks like:
Array
(
    [0] => phptest.php
    [1] => pi.php
    [2] => post_output.php
    [3] => test.php
    [4] => log.txt
    [5] => test.txt
)
*/
 



請註意,這些文件其實是可以返回一個路徑,這取決於查詢條件:


以下為引用的內容:


$files = glob(../images/a*.jpg);


print_r($files);
/* output looks like:
Array
(
    [0] => ../images/apple.jpg
    [1] => ../images/art.jpg
)
*/
 



如果你想獲得每個文件的完整路徑,你可以調用 realpath() 函數:


以下為引用的內容:


$files = glob(../images/a*.jpg);


// applies the function to each array element
$files = array_map(realpath,$files);


print_r($files);
/* output looks like:
Array
(
    [0] => C:wampwwwimagesapple.jpg
    [1] => C:wampwwwimagesart.jpg
)
*/
 


3、內存使用信息


通過偵測腳本的內存使用情況,有利於代碼的優化。PHP 提供瞭一個垃圾收集器和一個非常復雜的內存管理器。腳本執行時所使用的內存量,有升有跌。為瞭得到當前的內存使用情況,我們可以使用 memory_get_usage() 函數。如果需要獲得任意時間點的最高內存使用量,則可以使用 memory_limit() 函數。


以下為引用的內容:


echo “Initial: “.memory_get_usage().” bytes
“;
/* prints
Initial: 361400 bytes
*/


// lets use up some memory
for ($i = 0; $i < 100000; $i++) {
 $array []= md5($i);
}


// lets remove half of the array
for ($i = 0; $i < 100000; $i++) {
 unset($array[$i]);
}


echo “Final: “.memory_get_usage().” bytes
“;
/* prints
Final: 885912 bytes
*/


echo “Peak: “.memory_get_peak_usage().” bytes
“;
/* prints
Peak: 13687072 bytes
*/
 



4、CPU 使用信息


為此,我們要利用 getrusage() 函數。請記住這個函數不適用於 Windows 平臺。


以下為引用的內容:


print_r(getrusage());
/* prints
Array
(
    [ru_oublock] => 0
    [ru_inblock] => 0
    [ru_msgsnd] => 2
    [ru_msgrcv] => 3
    [ru_maxrss] => 12692
    [ru_ixrss] => 764
    [ru_idrss] => 3864
    [ru_minflt] => 94
    [ru_majflt] => 0
    [ru_nsignals] => 1
    [ru_nvcsw] => 67
    [ru_nivcsw] => 4
    [ru_nswap] => 0
    [ru_utime.tv_usec] => 0
    [ru_utime.tv_sec] => 0
    [ru_stime.tv_usec] => 6269
    [ru_stime.tv_sec] => 0
)
 



*/這可能看起來有點神秘,除非你已經有系統管理員權限。以下是每個值的具體說明(你不需要記住這些):


以下為引用的內容:


ru_oublock: block output operations
ru_inblock: block input operations
ru_msgsnd: messages sent
ru_msgrcv: messages received
ru_maxrss: maximum resident set size
ru_ixrss: integral shared memory size
ru_idrss: integral unshared data size
ru_minflt: page reclaims
ru_majflt: page faults
ru_nsignals: signals received
ru_nvcsw: voluntary context switches
ru_nivcsw: involuntary context switches
ru_nswap: swaps
ru_utime.tv_usec: user time used (microseconds)
ru_utime.tv_sec: user time used (seconds)
ru_stime.tv_usec: system time used (microseconds)
ru_stime.tv_sec: system time used (seconds)
 



要知道腳本消耗多少 CPU 功率,我們需要看看 ‘user time’ 和 ’system time’ 兩個參數的值。秒和微秒部分默認是單獨提供的。你可以除以 100 萬微秒,並加上秒的參數值,得到一個十進制的總秒數。讓我們來看一個例子:


以下為引用的內容:


// sleep for 3 seconds (non-busy)
sleep(3);


$data = getrusage();
echo “User time: “.
 ($data[ru_utime.tv_sec] +
 $data[ru_utime.tv_usec] / 1000000);
echo “System time: “.
 ($data[ru_stime.tv_sec] +
 $data[ru_stime.tv_usec] / 1000000);


/* prints
User time: 0.011552
System time: 0
*/
 



盡管腳本運行用瞭大約 3 秒鐘,CPU 使用率卻非常非常低。因為在睡眠運行的過程中,該腳本實際上不消耗 CPU 資源。還有許多其他的任務,可能需要一段時間,但不占用類似等待磁盤操作等 CPU 時間。因此正如你所看到的,CPU 使用率和運行時間的實際長度並不總是相同的。下面是一個例子:


以下為引用的內容:


// loop 10 million times (busy)
for($i=0;$i<10000000;$i++) {


}


$data = getrusage();
echo “User time: “.
 ($data[ru_utime.tv_sec] +
 $data[ru_utime.tv_usec] / 1000000);
echo “System time: “.
 ($data[ru_stime.tv_sec] +
 $data[ru_stime.tv_usec] / 1000000);


/* prints
User time: 1.424592
System time: 0.004204
*/
 



這花瞭大約 1.4 秒的 CPU 時間,但幾乎都是用戶時間,因為沒有系統調用。系統時間是指花費在執行程序的系統調用時的 CPU 開銷。下面是一個例子:


以下為引用的內容:


$start = microtime(true);
// keep calling microtime for about 3 seconds
while(microtime(true) – $start < 3) {


}


$data = getrusage();
echo “User time: “.
 ($data[ru_utime.tv_sec] +
 $data[ru_utime.tv_usec] / 1000000);
echo “System time: “.
 ($data[ru_stime.tv_sec] +
 $data[ru_stime.tv_usec] / 1000000);


/* prints
User time: 1.088171
System time: 1.675315
*/
 



現在我們有相當多的系統時間占用。這是因為腳本多次調用 microtime() 函數,該函數需要向操作系統發出請求,以獲取所需時間。你也可能會註意到運行時間加起來不到 3 秒。這是因為有可能在服務器上同時存在其他進程,並且腳本沒有 100% 使用 CPU 的整個 3 秒持續時間。


5、魔術常量


PHP 提供瞭獲取當前行號 (__LINE__)、文件路徑 (

發佈留言