设为首页 收藏本站
查看: 571|回复: 0

[经验分享] 9个你需要知道的PHP函数和功能

[复制链接]

尚未签到

发表于 2017-4-8 10:02:12 | 显示全部楼层 |阅读模式
即使使用 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.";

运维网声明 1、欢迎大家加入本站运维交流群:群②:261659950 群⑤:202807635 群⑦870801961 群⑧679858003
2、本站所有主题由该帖子作者发表,该帖子作者与运维网享有帖子相关版权
3、所有作品的著作权均归原作者享有,请您和我们一样尊重他人的著作权等合法权益。如果您对作品感到满意,请购买正版
4、禁止制作、复制、发布和传播具有反动、淫秽、色情、暴力、凶杀等内容的信息,一经发现立即删除。若您因此触犯法律,一切后果自负,我们对此不承担任何责任
5、所有资源均系网友上传或者通过网络收集,我们仅提供一个展示、介绍、观摩学习的平台,我们不对其内容的准确性、可靠性、正当性、安全性、合法性等负责,亦不承担任何法律责任
6、所有作品仅供您个人学习、研究或欣赏,不得用于商业或者其他用途,否则,一切后果均由您自己承担,我们对此不承担任何法律责任
7、如涉及侵犯版权等问题,请您及时通知我们,我们将立即采取措施予以解决
8、联系人Email:admin@iyunv.com 网址:www.yunweiku.com

所有资源均系网友上传或者通过网络收集,我们仅提供一个展示、介绍、观摩学习的平台,我们不对其承担任何法律责任,如涉及侵犯版权等问题,请您及时通知我们,我们将立即处理,联系人Email:kefu@iyunv.com,QQ:1061981298 本贴地址:https://www.yunweiku.com/thread-361814-1-1.html 上篇帖子: [转]优化PHP代码的40条建议 下篇帖子: ASP/PHP/ASP.net生成静态页大全
您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

扫码加入运维网微信交流群X

扫码加入运维网微信交流群

扫描二维码加入运维网微信交流群,最新一手资源尽在官方微信交流群!快快加入我们吧...

扫描微信二维码查看详情

客服E-mail:kefu@iyunv.com 客服QQ:1061981298


QQ群⑦:运维网交流群⑦ QQ群⑧:运维网交流群⑧ k8s群:运维网kubernetes交流群


提醒:禁止发布任何违反国家法律、法规的言论与图片等内容;本站内容均来自个人观点与网络等信息,非本站认同之观点.


本站大部分资源是网友从网上搜集分享而来,其版权均归原作者及其网站所有,我们尊重他人的合法权益,如有内容侵犯您的合法权益,请及时与我们联系进行核实删除!



合作伙伴: 青云cloud

快速回复 返回顶部 返回列表