Heim  >  Artikel  >  Backend-Entwicklung  >  九个你需要知道的PHP函数和功能

九个你需要知道的PHP函数和功能

WBOY
WBOYOriginal
2016-06-13 13:02:59812Durchsuche

9个你需要知道的PHP函数和功能
即使使用 PHP 多年,有些功能和特点我们也未必发现或未被充分利用,一旦被我们发现,就会发现它们非常有用。然而,并不是所有的人都已经从头至尾详读过 PHP 的手册和功能参考!

1. 函数与任意数量的参数
您可能已经知道,PHP 允许我们定义可选参数的函数。但也有完全允许任意数量的函数参数方法。

首先,下面这个例子只是可选参数:

view sourceprint?01 // function with 2 optional arguments 

02 function foo($arg1 = '', $arg2 = '') { 

03     echo "arg1: $arg1\n"; 

04     echo "arg2: $arg2\n"; 

05 } 

06 foo('hello','world'); 

07 /* prints: 

08 arg1: hello 

09 arg2: world 

10 */

11 foo(); 

12 /* prints: 

13 arg1: 

14 arg2: 

15 */

现在,让我们看看如何可以建立一个函数接受任何数量的参数。这一次,我们要利用 func_get_args() 函数:

view sourceprint?01 // yes, the argument list can be empty 

02 function foo() { 

03     // returns an array of all passed arguments 

04     $args = func_get_args(); 

05     foreach ($args as $k => $v) { 

06         echo "arg".($k+1).": $v\n"; 

07     } 

08 } 

09 foo(); 

10 /* prints nothing */

11 foo('hello'); 

12 /* prints 

13 arg1: hello 

14 */

15 foo('hello', 'world', 'again'); 

16 /* prints 

17 arg1: hello 

18 arg2: world 

19 arg3: again 

20 */

2. 使用 Glob() 函数来查找文件
许多 PHP 内置函数有非常长的命名。然而,它可能会很难说明是什么作用的函数,如果不使用 Glob() 来做,除非你已经非常熟悉这个函数。

它更像是 scandir() 函数加强型版本。它可以让您通过使用模式搜索文件。

view sourceprint?01 // get all php files 

02 $files = glob('*.php'); 

03 print_r($files); 

04 /* output looks like: 

05 Array 

06 ( 

07     [0] => phptest.php 

08     [1] => pi.php 

09     [2] => post_output.php 

10     [3] => test.php 

11 ) 

12 */

这样你可以获得多个文件类型:

view sourceprint?01 // get all php files AND txt files 

02 $files = glob('*.{php,txt}', GLOB_BRACE); 

03 print_r($files); 

04 /* output looks like: 

05 Array 

06 ( 

07     [0] => phptest.php 

08     [1] => pi.php 

09     [2] => post_output.php 

10     [3] => test.php 

11     [4] => log.txt 

12     [5] => test.txt 

13 ) 

14 */

请注意,这些文件其实是可以返回一个路径的,根据你的查询。

view sourceprint?1 $files = glob('../images/a*.jpg'); 

2 print_r($files); 

3 /* output looks like: 

4 Array 

5 ( 

6     [0] => ../images/apple.jpg 

7     [1] => ../images/art.jpg 

8 ) 

9 */

如果你想获得每个文件的完整路径,你可以调用 realpath() 函数来返回。

view sourceprint?01 $files = glob('../images/a*.jpg'); 

02 // applies the function to each array element 

03 $files = array_map('realpath',$files); 

04 print_r($files); 

05 /* output looks like: 

06 Array 

07 ( 

08     [0] => C:\wamp\www\images\apple.jpg 

09     [1] => C:\wamp\www\images\art.jpg 

10 ) 

11 */

3. 内存使用信息
通过观察你的脚本内存使用情况,你就可以将你的代码进行针对性优化。

PHP 有一个垃圾收集器和一个相当复杂的内存管理器。当你的脚本开始就开始正式使用内存。也会根据脚本的执行情况,内存使用量会上升也会下降。为了得到当前内存使用情况,我们就可以使用 memory_get_usage() 函数。并可以在任何时候得到内存使用的最高点,下面就是使用 memory_get_usage() 函数的例子。

view sourceprint?01 echo "Initial: ".memory_get_usage()." bytes \n"; 

02 /* prints 

03 Initial: 361400 bytes 

04 */

05 // let's use up some memory 

06 for ($i = 0; $i
07     $array []= md5($i); 

08 } 

09 // let's remove half of the array 

10 for ($i = 0; $i
11     unset($array[$i]); 

12 } 

13 echo "Final: ".memory_get_usage()." bytes \n"; 

14 /* prints 

15 Final: 885912 bytes 

16 */

17 echo "Peak: ".memory_get_peak_usage()." bytes \n"; 

18 /* prints 

19 Peak: 13687072 bytes 

20 */

4. CPU 的使用信息
为此,我们就要利用 getrusage() 函数。请记住,这个函数不能应用于 windows 平台。

view sourceprint?01 print_r(getrusage()); 

02 /* prints 

03 Array 

04 ( 

05     [ru_oublock] => 0 

06     [ru_inblock] => 0 

07     [ru_msgsnd] => 2 

08     [ru_msgrcv] => 3 

09     [ru_maxrss] => 12692 

10     [ru_ixrss] => 764 

11     [ru_idrss] => 3864 

12     [ru_minflt] => 94 

13     [ru_majflt] => 0 

14     [ru_nsignals] => 1 

15     [ru_nvcsw] => 67 

16     [ru_nivcsw] => 4 

17     [ru_nswap] => 0 

18     [ru_utime.tv_usec] => 0 

19     [ru_utime.tv_sec] => 0 

20     [ru_stime.tv_usec] => 6269 

21     [ru_stime.tv_sec] => 0 

22 ) 

23 */

这看起来蛮神秘的,有些艰涩难懂,除非你已经有过系统管理员的经验,以下是每个值的介绍(或许你并不需要记住这些):

ru_oublock:块输出操作
ru_inblock:块输入操作
ru_msgsnd:邮件发送
ru_msgrcv:收到的邮件
ru_maxrss:最大驻留集大小
ru_ixrss:积分共享内存的大小
ru_idrss:积分大小非共享数据
ru_minflt:页回收
ru_majflt:页面错误
ru_nsignals:信号接收
ru_nvcsw:自动上下文切换
ru_nivcsw:非自动的上下文切换
ru_nswap:过期
ru_utime.tv_usec:用户使用时间(微秒)
ru_utime.tv_sec:用户使用时间(秒)
ru_stime.tv_usec:系统使用时间(微秒)
ru_stime.tv_sec:系统使用时间(秒)
要看 CPU 的功率有多少被脚本消耗,我们需要观察 user time 和 system time 的值。秒和毫秒是默认独立提供的。你可以将 100 万毫秒的值,并将其换算成秒的值,将它当做一个十进制数的总秒数。

让我们看一个例子:

view sourceprint?01 // sleep for 3 seconds (non-busy) 

02 sleep(3); 

03 $data = getrusage(); 

04 echo "User time: ". 

05     ($data['ru_utime.tv_sec'] + 

06     $data['ru_utime.tv_usec'] / 1000000); 

07 echo "System time: ". 

08     ($data['ru_stime.tv_sec'] + 

09     $data['ru_stime.tv_usec'] / 1000000); 

10 /* prints 

11 User time: 0.011552 

12 System time: 0 

13 */

虽然脚本大约花了 3 秒钟的时间来运行, CPU 的使用率还是非常非常低的。因为 sleep 工作,脚本实际上并没有消耗 CPU 资源。当然还有其他的任务可能真正需要等待时间,但千万不能用磁盘的读取写入操作来等待 CPU 时间。所以你可以发现, CPU 使用率和运行时的实际长度并不是总是一样的。

下面是另外一个例子。

view sourceprint?01 // loop 10 million times (busy) 

02 for($i=0;$i
03 } 

04 $data = getrusage(); 

05 echo "User time: ". 

06     ($data['ru_utime.tv_sec'] + 

07     $data['ru_utime.tv_usec'] / 1000000); 

08 echo "System time: ". 

09     ($data['ru_stime.tv_sec'] + 

10     $data['ru_stime.tv_usec'] / 1000000); 

11 /* prints 

12 User time: 1.424592 

13 System time: 0.004204 

14 */

这花了大约 1.4 秒的 CPU 时间。几乎所有这些都是由用户操作所用的时间,系统并没有被调用。

系统时间是划分时间的 CPU 上执行的程序的代表的内核系统调用时间。(谁有更简明扼要的描述?help!)下面是一个例子:

view sourceprint?01 $start = microtime(true); 

02 // keep calling microtime for about 3 seconds 

03 while(microtime(true) - $start
04 } 

05 $data = getrusage(); 

06 echo "User time: ". 

07     ($data['ru_utime.tv_sec'] + 

08     $data['ru_utime.tv_usec'] / 1000000); 

09 echo "System time: ". 

10     ($data['ru_stime.tv_sec'] + 

11     $data['ru_stime.tv_usec'] / 1000000); 

12 /* prints 

13 User time: 1.088171 

14 System time: 1.675315 

15 */

5. 魔术常量
PHP 提供了获取当前行号的方法 (__LINE__),获取文件路径方法(__FILE__),目录(__DIR__),函数名(__FUNCTTION__),类名(__CLASS__),方法名(__METHOD__),和命名空间(__NAMESPACE__)。以上就是常用的魔术常量。恐怕我们最常用的就只有 (__FILE__) 了。

Rikku 不打算全部进行说明,但会说几个用例。

当然包括了其他的脚本,这是个不错的主意。((__DIR__)需要 PHP 5.3 以上版本):

view sourceprint?1 // this is relative to the loaded script's path 

2 // it may cause problems when running scripts from different directories 

3 require_once('config/database.php'); 

4 // this is always relative to this file's path 

5 // no matter where it was included from 

6 require_once(dirname(__FILE__) . '/config/database.php');

使用 __LINE__ 让调试更加容易,你可以跟踪行号:

view sourceprint?01 // some code 

02 // ... 

03 my_debug("some debug message", __LINE__); 

04 /* prints 

05 Line 4: some debug message 

06 */

07 // some more code 

08 // ... 

09 my_debug("another debug message", __LINE__); 

10 /* prints 

11 Line 11: another debug message 

12 */

13 function my_debug($msg, $line) { 

14     echo "Line $line: $msg\n"; 

15 }

6. 生成唯一的ID
有些情况下,您需要生成一股唯一的字符串。我看到很多人会用这个 md5() 函数,即使他并不完全用于此目的的存在:

view sourceprint?1 // generate unique string 

2 echo md5(time() . mt_rand(1,1000000));

其实有个专门的 PHP 函数,名为 uniqid() 就是为了这个目的而存在的:

view sourceprint?01 // generate unique string 

02 echo uniqid(); 

03 /* prints 

04 4bd67c947233e 

05 */

06 // generate another unique string 

07 echo uniqid(); 

08 /* prints 

09 4bd67c9472340 

10 */

您可能会注意到,即使是唯一的字符串,他们的前几个字符很相似。这是因为生成的字符串是关联到服务器时间的。设实际上有一个非常好的副作用,因为每个新生成的 ID 将在生成后按字母顺序排列,这样也省去了我们排序的逻辑动作。

为了减少重复的几率,你可以传递一个前缀,或在第二个参数来增加。

view sourceprint?01 // with prefix 

02 echo uniqid('foo_'); 

03 /* prints 

04 foo_4bd67d6cd8b8f 

05 */

06 // with more entropy 

07 echo uniqid('',true); 

08 /* prints 

09 4bd67d6cd8b926.12135106 

10 */

11 // both 

12 echo uniqid('bar_',true); 

13 /* prints 

14 bar_4bd67da367b650.43684647 

15 */

此功能将会生成比 md5() 生成的字符串更短,这也将节省您的空间。

7. 序列化
你有没有需要存储在数据库中复杂的变量或者大文本文件?那你有没有拿出一个解决方法,花式转换成格式化的字符串数组或对象的?别担心,PHP 已经为我们准备好了这个功能。

有两种序列化的方法,下面是一个例子,它使用 serialize() 进行序列化和 unserialize() 进行解除序列化:

view sourceprint?01 // a complex array 

02 $myvar = array( 

03     'hello', 

04     42, 

05     array(1,'two'), 

06     'apple'

07 ); 

08 // convert to a string 

09 $string = serialize($myvar); 

10 echo $string; 

11 /* prints 

12 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";} 

13 */

14 // you can reproduce the original variable 

15 $newvar = unserialize($string); 

16 print_r($newvar); 

17 /* prints 

18 Array 

19 ( 

20     [0] => hello 

21     [1] => 42 

22     [2] => Array 

23         ( 

24             [0] => 1 

25             [1] => two 

26         ) 

27     [3] => apple 

28 ) 

29 */

这是原生态的 PHP 序列化方法。然而,由于 JSON 近年来已经大受欢迎,PHP 5.2 中也决定添加对它们的支持。现在你可以使用 json_encode() 和 json_decode() 函数来完成这项工作:

view sourceprint?01 // a complex array 

02 $myvar = array( 

03     'hello', 

04     42, 

05     array(1,'two'), 

06     'apple'

07 ); 

08 // convert to a string 

09 $string = json_encode($myvar); 

10 echo $string; 

11 /* prints 

12 ["hello",42,[1,"two"],"apple"] 

13 */

14 // you can reproduce the original variable 

15 $newvar = json_decode($string); 

16 print_r($newvar); 

17 /* prints 

18 Array 

19 ( 

20     [0] => hello 

21     [1] => 42 

22     [2] => Array 

23         ( 

24             [0] => 1 

25             [1] => two 

26         ) 

27     [3] => apple 

28 ) 

29 */

这么做看起来会更加紧凑。当然它对其他语言如 javascript 兼容性也是最好的。然而,您需要注意的是:对于某些复杂的对象,某些信息会无故丢失!

8. 压缩字符串
在谈到压缩时,我们通常会想到一些文件,如 zip 文件。它可以在 PHP 中压缩长字符串,并且不涉及任何存档文件。

在下面的例子中,我们要利用 gzcompress() 和 gzuncompress() 函数:

view sourceprint?01 $string = 

02 "Lorem ipsum dolor sit amet, consectetur 

03 adipiscing elit. Nunc ut elit id mi ultricies 

04 adipiscing. Nulla facilisi. Praesent pulvinar, 

05 sapien vel feugiat vestibulum, nulla dui pretium orci, 

06 non ultricies elit lacus quis ante. Lorem ipsum dolor 

07 sit amet, consectetur adipiscing elit. Aliquam 

08 pretium ullamcorper urna quis iaculis. Etiam ac massa 

09 sed turpis tempor luctus. Curabitur sed nibh eu elit 

10 mollis congue. Praesent ipsum diam, consectetur vitae 

11 ornare a, aliquam a nunc. In id magna pellentesque 

12 tellus posuere adipiscing. Sed non mi metus, at lacinia 

13 augue. Sed magna nisi, ornare in mollis in, mollis 

14 sed nunc. Etiam at justo in leo congue mollis. 

15 Nullam in neque eget metus hendrerit scelerisque 

16 eu non enim. Ut malesuada lacus eu nulla bibendum 

17 id euismod urna sodales. "; 

18 $compressed = gzcompress($string); 

19 echo "Original size: ". strlen($string)."\n"; 

20 /* prints 

21 Original size: 800 

22 */

23 echo "Compressed size: ". strlen($compressed)."\n"; 

24 /* prints 

25 Compressed size: 418 

26 */

27 // getting it back 

28 $original = gzuncompress($compressed);

我们能够压缩近 50% 。另外 gzencode() 和 gzdecode() 可以达成类似的结果,但通过的是不同的压缩算法。

9. register_shutdown_function
有一个函数叫 register_shutdown_function(),可以让你在拥有执行一些代码权限之前,完成脚本的运行。

试想一下,你想捕捉到你脚本执行至结束时一些基准的统计数据,如一共用了多少时间来执行:

view sourceprint?1 // capture the start time 

2 $start_time = microtime(true); 

3 // do some stuff 

4 // ... 

5 // display how long the script took 

6 echo "execution took: ". 

7         (microtime(true) - $start_time). 

8         " seconds.";

起初觉得这些似乎是微不足道的。你只要添加代码放在底部,它运行脚本之前完成。不过,在脚本程序其中你调用了 exit() 函数,那么该段代码将不被执行。此外,如果有一个致命的错误,或者该脚本由用户终止(就是按浏览器上面的停止按钮),再次刷新页面也无法被运行。

当您使用 register_shutdown_function(),你的代码将没有理由被迫停止:

view sourceprint?01 $start_time = microtime(true); 

02 register_shutdown_function('my_shutdown'); 

03 // do some stuff 

04 // ... 

05 function my_shutdown() { 

06     global $start_time; 

07     echo "execution took: ". 

08             (microtime(true) - $start_time). 

09             " seconds."; 

Stellungnahme:
Der Inhalt dieses Artikels wird freiwillig von Internetnutzern beigesteuert und das Urheberrecht liegt beim ursprünglichen Autor. Diese Website übernimmt keine entsprechende rechtliche Verantwortung. Wenn Sie Inhalte finden, bei denen der Verdacht eines Plagiats oder einer Rechtsverletzung besteht, wenden Sie sich bitte an admin@php.cn