当前位置:Gxlcms > PHP教程 > 九个你需要知道的PHP函数和功能

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

时间:2021-07-01 10:21:17 帮助过:24人阅读

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 < 100000; $i++) {

07 $array []= md5($i);

08 }

09 // let's remove half of the array

10 for ($i = 0; $i < 100000; $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< 10000000;$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 < 3) {

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.";

人气教程排行