做過PHP開發的程序員應該清楚,PHP中有很多內置的功能,掌握了它們,可以幫助你在做PHP開發時更加得心應手,本文將分享8個開發必備的PHP功能,個個都非常實用,希望各位PHP開發者能夠掌握。
我們在.NET或者JAVA編程中,一般函數參數個數都是固定的,但是PHP允許你使用任意個數的參數。下面這個示例向你展示了PHP函數的默認參數:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19// 兩個默認參數的函數
function
foo(
$arg1
= ”,
$arg2
= ”) {
echo
“arg1:
$arg1
\n”;
echo
“arg2:
$arg2
\n”;
}
foo(‘hello’,'world’);
/* 輸出:
arg1: hello
arg2: world
*/
foo();
/* 輸出:
arg1:
arg2:
*/
下面這個示例是PHP的不定參數用法,其使用到了?func_get_args()方法:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26// 是的,形參列表為空
function
foo() {
// 取得所有的傳入參數的數組
$args
= func_get_args();
foreach
(
$args
as
$k
=>
$v
) {
echo
“arg”.(
$k
+1).”:
$v
\n”;
}
}
foo();
/* 什麼也不會輸出 */
foo(‘hello’);
/* 輸出
arg1: hello
*/
foo(‘hello’, ‘world’, ‘again’);
/* 輸出
arg1: hello
arg2: world
arg3: again
*/
大部分PHP函數的函數名從字面上都可以理解其用途,但是當你看到?glob() 的時候,你也許並不知道這是用來做什麼的,其實glob()和scandir() 一樣,可以用來查找文件,請看下面的用法:
1 2 3 4 5 6 7 8 9 10 11 12 13// 取得所有的後綴為PHP的文件
$files
=
glob
(‘*.php’);
print_r(
$files
);
/* 輸出:
Array
(
[0] => phptest.php
[1] => pi.php
[2] => post_output.php
[3] => test.php
)
*/
你還可以查找多種後綴名
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15// 取PHP文件和TXT文件
$files
=
glob
(‘*.{php,txt}’, GLOB_BRACE);
print_r(
$files
);
/* 輸出:
Array
(
[0] => phptest.php
[1] => pi.php
[2] => post_output.php
[3] => test.php
[4] => log.txt
[5] => test.txt
)
*/
你還可以加上路徑:
1 2 3 4 5 6 7 8 9 10$files
=
glob
(‘../images/a*.jpg’);
print_r(
$files
);
/* 輸出:
Array
(
[0] => ../images/apple.jpg
[1] => ../images/art.jpg
)
*/
如果你想得到絕對路徑,你可以調用?realpath() 函數:
1 2 3 4 5 6 7 8 9 10 11 12 13$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:\wamp\www\images\apple.jpg
[1] => C:\wamp\www\images\art.jpg
)
*/
PHP的內存回收機制已經非常強大,你也可以使用PHP腳本獲取當前內存的使用情況,調用memory_get_usage() 函數獲取當期內存使用情況,調用memory_get_peak_usage() 函數獲取內存使用的峰值。參考代碼如下:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24echo
“Initial: “.memory_get_usage().” bytes \n”;
/* 輸出
Initial: 361400 bytes
*/
// 使用內存
for
(
$i
= 0;
$i
< 100000;
$i
++) {
$array
[]= md5(
$i
);
}
// 刪除一半的內存
for
(
$i
= 0;
$i
< 100000;
$i
++) {
unset(
$array
[
$i
]);
}
echo
“Final: “.memory_get_usage().” bytes \n”;
/* prints
Final: 885912 bytes
*/
echo
“Peak: “.memory_get_peak_usage().” bytes \n”;
/* 輸出峰值
Peak: 13687072 bytes
*/
獲取了內存使用情況,也可以使用PHP的getrusage()獲取CPU使用情況,該方法在windows下不可用。
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24print_r(
getrusage
());
/* 輸出
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
)
*/
這個結構看上出很晦澀,除非你對CPU很了解。下面一些解釋:
要看到你的腳本消耗了多少CPU,我們需要看看“用戶態的時間”和“系統內核時間”的值。秒和微秒部分是分別提供的,您可以把微秒值除以100萬,並把它添加到秒的值後,可以得到有小數部分的秒數。
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15// 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);
/* 輸出
User time: 0.011552
System time: 0
*/
sleep是不占用系統時間的,我們可以來看下面的一個例子:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17// 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);
/* 輸出
User time: 1.424592
System time: 0.004204
*/
這花了大約14秒的CPU時間,幾乎所有的都是用戶的時間,因為沒有系統調用。
系統時間是CPU花費在系統調用上的上執行內核指令的時間。下面是一個例子:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18$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
*/
我們可以看到上面這個例子更耗CPU。
PHP 提供非常有用的系統常量 可以讓你得到當前的行號 (__LINE__),文件 (__FILE__),目錄 (__DIR__),函數名 (__FUNCTION__),類名(__CLASS__),方法名(__METHOD__) 和名字空間 (__NAMESPACE__),很像C語言。
我們可以以為這些東西主要是用於調試,當也不一定,比如我們可以在include其它文件的時候使用?__FILE__ (當然,你也可以在 PHP 5.3以後使用 __DIR__ ),下面是一個例子。
1 2 3 4 5 6 7// this is relative to the loaded script’s path
// it may cause problems when running scripts from different directories
require_once
(‘config/database.php’);
// this is always relative to this file’s path
// no matter where it was included from
require_once
(dirname(
__FILE__
) . ‘/config/database.php’);
下面是使用 __LINE__ 來輸出一些debug的信息,這樣有助於你調試程序:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17// some code
// …
my_debug(“some debug message”,
__LINE__
);
/* 輸出
Line 4: some debug message
*/
// some more code
// …
my_debug(“another debug message”,
__LINE__
);
/* 輸出
Line 11: another debug message
*/
function
my_debug(
$msg
,
$line
) {
echo
“Line
$line
:
$msg
\n”;
}
很多朋友都利用md5()來生成唯一的編號,但是md5()有幾個缺點:1、無序,導致數據庫中排序性能下降。2、太長,需要更多的存儲空間。其實PHP中自帶一個函數來生成唯一的id,這個函數就是uniqid()。下面是用法:
1 2 3 4 5 6 7 8 9 10 11// generate unique string
echo
uniqid();
/* 輸出
4bd67c947233e
*/
// generate another unique string
echo
uniqid();
/* 輸出
4bd67c9472340
*/
該算法是根據CPU時間戳來生成的,所以在相近的時間段內,id前幾位是一樣的,這也方便id的排序,如果你想更好的避免重復,可以在id前加上前綴,如:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17// 前綴
echo
uniqid(‘foo_’);
/* 輸出
foo_4bd67d6cd8b8f
*/
// 有更多的熵
echo
uniqid(”,true);
/* 輸出
4bd67d6cd8b926.12135106
*/
// 都有
echo
uniqid(‘bar_’,true);
/* 輸出
bar_4bd67da367b650.43684647
*/
PHP序列化功能大家可能用的比較多,也比較常見,當你需要把數據存到數據庫或者文件中是,你可以利用PHP中的serialize() 和 unserialize()方法來實現序列化和反序列化,代碼如下:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34// 一個復雜的數組
$myvar
=
array
(
‘hello’,
42,
array
(1,’two’),
‘apple’
);
// 序列化
$string
= serialize(
$myvar
);
echo
$string
;
/* 輸出
a:4:{i:0;s:5:”hello”;i:1;i:42;i:2;a:2:{i:0;i:1;i:1;s:3:”two”;}i:3;s:5:”apple”;}
*/
// 反序例化
$newvar
= unserialize(
$string
);
print_r(
$newvar
);
/* 輸出
Array
(
[0] => hello
[1] => 42
[2] => Array
(
[0] => 1
[1] => two
)
[3] => apple
)
*/
如何序列化成json格式呢,放心,php也已經為你做好了,使用php 5.2以上版本的用戶可以使用json_encode() 和 json_decode() 函數來實現json格式的序列化,代碼如下:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34// a complex array
$myvar
=
array
(
‘hello’,
42,
array
(1,’two’),
‘apple’
);
// convert to a string
$string
= json_encode(
$myvar
);
echo
$string
;
/* prints
["hello",42,[1,"two"],”apple”]
*/
// you can reproduce the original variable
$newvar
= json_decode(
$string
);
print_r(
$newvar
);
/* prints
Array
(
[0] => hello
[1] => 42
[2] => Array
(
[0] => 1
[1] => two
)
[3] => apple
)
*/
當我們說到壓縮,我們可能會想到文件壓縮,其實,字符串也是可以壓縮的。PHP提供了?gzcompress() 和 gzuncompress() 函數:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32$string
=
“Lorem ipsum dolor sit amet, consectetur
adipiscing elit. Nunc ut elit id mi ultricies
adipiscing. Nulla facilisi. Praesent pulvinar,
sapien vel feugiat vestibulum, nulla dui pretium orci,
non ultricies elit lacus quis ante. Lorem ipsum dolor
sit amet, consectetur adipiscing elit. Aliquam
pretium ullamcorper urna quis iaculis. Etiam ac massa
sed turpis tempor luctus. Curabitur sed nibh eu elit
mollis congue. Praesent ipsum diam, consectetur vitae
ornare a, aliquam a nunc. In id magna pellentesque
tellus posuere adipiscing. Sed non mi metus, at lacinia
augue. Sed magna nisi, ornare in mollis in, mollis
sed nunc. Etiam at justo in leo congue mollis.
Nullam in neque eget metus hendrerit scelerisque
eu non enim. Ut malesuada lacus eu nulla bibendum
id euismod urna sodales. “;
$compressed
= gzcompress(
$string
);
echo
“Original size: “.
strlen
(
$string
).”\n”;
/* 輸出原始大小
Original size: 800
*/
echo
“Compressed size: “.
strlen
(
$compressed
).”\n”;
/* 輸出壓縮後的大小
Compressed size: 418
*/
// 解壓縮
$original
= gzuncompress(
$compressed
);
幾乎有50% 壓縮比率。同時,你還可以使用?gzencode() 和 gzdecode() 函數來壓縮,只不用其用了不同的壓縮算法。
以上就是8個開發必備的PHP功能,是不是都很實用呢?