chop 是rtrim()的别名;
ltrim()
trim()
nl2br()将\n转换成<br>
print,echo,printf(),sprintf():
echo()不是函数,print()是函数,有返回值,boolen,false,true;
printf()格式化输出
--函数,把文字格式化以后输出,直接调用系统调用进行IO的,他是非缓冲的。如:
$name="hunte";
$age=25;
printf("my name is %s, age %d", $name, $age);
sprintf()格式化字符串,然后赋给一个变量,但是不输出,类似于c了
<?php
echo nl2br("foo isn't\n bar");
echo "foo isn't\n bar";
?>
--跟printf相似,但不打印,而是返回格式化后的文字,其他的与printf一样。如:
char sql[256];
sprintf(sql,"select * from table where no = '%s'",bankno);
它的功能只是把""里面的语句赋给了变量sql。
strtolower
strtoupper
ucwords
ucfirst
2.字符串的连接和分割
(1)array explode(string input ,string separator , int limit)
使用一个字符串分割另一个字符串
<?php
// 示例 1
$pizza = "piece1 piece2 piece3 piece4 piece5 piece6";
$pieces = explode(" ", $pizza);
echo $pieces[0]; // piece1
echo $pieces[1]; // piece2
// 示例 2
$data = "foo:*:1023:1000::/home/foo:/bin/sh";
list($user, $pass, $uid, $gid, $gecos, $home, $shell) = explode(":", $data);
echo $user; // foo
echo $pass; // *
?>
例子 2. limit 参数示例
<?php
$str = 'one|two|three|four';
// 正数的 limit
print_r(explode('|', $str, 2));
// 负数的 limit
print_r(explode('|', $str, -1));
?>
string strtok( string input ,string separator)
<?php
$string = "This is\tan example\nstring";
/* Use tab and newline as tokenizing characters as well */
$tok = strtok($string, " \n\t");
//拿着空格,\n,\t作为令牌分割字符串
while ($tok !== false) {
echo "Word=$tok<br />";
$tok = strtok(" \n\t");
}
?>
结果:
Word=This
Word=is
Word=an
Word=example
Word=string
(2.)字符串的截取
$test="Your customer service is excellent";
echo substr($test,1);////////our customer service is excellent
echo "<br>";
echo substr($test,-9);//////从末尾起长度是9excellent
echo "<br>";
echo substr($test,0,4);////从0位置开始长度是4Your
echo "<br>";
echo substr($test,5,-13);/从第四个开始到倒数第13个字符customer service
echo "<br>";
$test="Your customer service is excellent";
echo substr($test,1);
echo "<br>";
echo substr($test,-11);
echo "<br>";
echo substr($test,0,6);
echo "<br>";
echo substr($test,5,-13);
echo "<br>";
our customer service is excellent
s excellent
Your c
customer service
(3)join()字符串的链接
3.字符串的查找
(1)string strstr ( string haystack, string needle ) 别名:strchr,stristr和strstr类似不同在于不区分大小写
strrchr()相反的,查找的是最后一次出现的字符串
第一次出现起的字符串
<?php
$email = 'user@example.com';
$domain = strstr($email, '@');
echo $domain; // prints @example.com
?>
$email = 'user@example.com';
$domain =strstr($email,'e');
$domain2 =strrchr($email,'e');//最后一次出现起的字符串
echo $domain;
echo "<br>";
echo $domain2;
er@example.com
e.com
(2)查找位置
int strpos(string str,string needle,[int offset]) 没有找到返回的是false
返回从offset开始在str中查找needle的位置
$eg:$t-'hello world';
echo strpos($t,'o',5);
//7 从第o开始,查找o这个变量的位置,结果为7
int strrpos()
5.替换
str_replace("%body%","blank","<body text='%body%'")
6 。大写小问题
Strpos
查找字符串中第一次出现的字符串的位置
Strrpos
查找字符串中某字符,继第一次之后的最先出现的位置。
strpos(stripos无大小写)
strrpos(strripos无大小写)
strstr
stristr(无大小写)
str_replace
str_ireplace(无大小写)
str_replace() 函数使用一个字符串替换字符串中的另一些字符。
语法
str_replace(find,replace,string,count)
参数 描述
find 必需。规定要查找的值。
replace 必需。规定替换 find 中的值的值。
string 必需。规定被搜索的字符串。
count 可选。一个变量,对替换数进行计数。
提示和注释
注释:该函数对大小写敏感。请使用 str_ireplace() 执行对大小写不敏感的搜索。
注释:该函数是二进制安全的。
例子 1
<?php
echo str_replace("world","John","Hello world!");
?>
输出:
Hello John!
例子 2
在本例中,我们将演示带有数组和 count 变量的 str_replace() 函数:
<?php
$arr = array("blue","red","green","yellow");
print_r(str_replace("red","pink",$arr,$i));
echo "Replacements: $i";
?>
输出:
Array
(
[0] => blue
[1] => pink
[2] => green
[3] => yellow
)
Replacements: 1
例子 3
<?php
$find = array("Hello","world");
$replace = array("B");
$arr = array("Hello","world","!");
print_r(str_replace($find,$replace,$arr));
?>
输出:
Array
(
[0] => B
[1] =>
[2] => !
)
漏洞相关函数:
<?php
$arr1 = Array(
'http://img./img/offer/29/24/70/20/29247020',
'http://img./img/offer/29/24/70/20/29247020-1',
'http://img./img/offer/29/24/70/20/29247020-2'
);
$arr2 = Array(
'http://localhost/root/ups/af48056fc4.jpg',
'http://localhost/root/ups/cf33240aa3.jpg',
'http://localhost/root/ups/c30e40419b.jpg'
);
$data = '
<img src="http://img./img/offer/29/24/70/20/29247020"/>
<img src="http://img./img/offer/29/24/70/20/29247020-1"/>
<img src="http://img./img/offer/29/24/70/20/29247020-2"/>';
$data = str_replace($arr1,$arr2,$data);
var_dump($data);
?>
替换后的结果是:
string(169) "<img src="/root/ups/af48056fc4.jpg"/><img src="/root/ups/af48056fc4.jpg"/><img src="/root/ups/af48056fc4.jpg"/>"str_replace 函数的声明大概是这样: str_replace($search, $replace, $input[,&$count]), 比如在对一个字符串进行替换操作, $input 就是源字符串(称为数据源). 这很不合理,因为它把数据源放在第3位, 而 str_pos, strtok, str_repeat 等等函数都是把数据源放在第1位.也就是说str_replace并没有替换掉数组中相对应的字符串,而是把数组中的第一个替换,然后把相同的字符串后多余的合并。
解决办法:
function strrplace($arr1,$arr2,$data){
if(is_array($arr1)) {
foreach($arr1 as $key => $value) {
$data = str_replace_once($value, $arr2[$key], $data);
} }
return $data;
}
function str_replace_once($needle, $replace, $data) //替换第一次
{
$pos = strpos($data, $needle);
if ($pos === false) {
return $data;
}
return substr_replace($data, $replace, $pos, strlen($needle));
}
1.在可以用file_get_contents替代file、fopen、feof、fgets等系列方法的情况下,尽量用file_get_contents,因为他的效率高得多!但是要注意file_get_contents在打开一个URL文件时候的PHP版本问题;
2.尽量的少进行文件操作,虽然PHP的文件操作效率也不低的;
3.优化SELECT SQL语句,在可能的情况下尽量少的进行INSERT、UPDATE操作(在UPDATE上,我被恶批过);
4.尽可能的使用PHP内部函数(但是我却为了找个PHP里面不存在的函数,浪费了本可以写出一个自定义函数的时间,经验问题啊!);
5.循环内部不要声明变量,尤其是大变量:对象(这好像不只是PHP里面要注意的问题吧?);
6.多维数组尽量不要循环嵌套赋值;
7.在可以用PHP内部字符串操作函数的情况下,不要用正则表达式;
8.foreach效率更高,尽量用foreach代替while和for循环;
9.用单引号替代双引号引用字符串;
10.用i+=1代替i=i+1。符合c/c++的习惯,效率还高;
11.对global变量,应该用完就unset()掉;
静态调用的成员一定要定义成static (PHP5 ONLY)
Tips: PHP5引入了静态成员的概念,作用和PHP4的函数内部静态变量一致,但前者是作为类的成员来使用。静态变量和Ruby的类变量(class variable)差不多,所有类的实例共享同一个静态变量。静态地调用非static 成员,效率会比静态地调用 static 成员慢 50-60%。主要是因为前者会产生 E_STRICT 警告,内部也需要做转换。
<?php
class foo {
function bar() {
echo 'foobar';
}
}
$foo = new foo;
// instance way
$foo->bar();
// static way
foo::bar();
?>
使用类常量(PHP5 ONLY)
Tips: PHP5新功能,类似于C++的const。
使用类常量的好处是:
- 编译时解析,没有额外开销
- 杂凑表更小,所以内部查找更快
- 类常量仅存在于特定「命名空间」,所以杂凑名更短
- 代码更干净,使除错更方便
(暂时)不要使用 require/include_once
require/include_once 每次被调用的时候都会打开目标文件!
- 如果用绝对路径的话,PHP 5.2/6.0 不存在这个问题
- 新版的 APC 缓存系统已经解决这个问题
文件 I/O 增加 => 效率降低
如果需要,可以自行检查文件是否已被 require/include。
不要调用毫无意义的函数
有对应的常量的时候,不要使用函数。
<?php
php_uname('s') == PHP_OS;
php_version() == PHP_VERSION;
php_sapi_name() == PHP_SAPI;
?>
虽然使用不多,但是效率提升大概在3500% 左右。
最快的 Win32 检查
<?php
$is_win = DIRECTORY_SEPARATOR == '\\';
?>
- 不用函数
- Win98/NT/2000/XP/Vista/Longhorn/Shorthorn/Whistler...通用
- 一直可用
时间问题 (PHP>5.1.0 ONLY)
你如何在你的软件中得知现在的时间?简单,「time() time() again, you ask me...」。
不过总归会调用函数,慢。
现在好了,用 $_SERVER['REQUEST_TIME'],不用调用函数,又省了。
加速 PCRE
- 对于不用保存的结果,不用 (),一律用(?:)
这样 PHP 不用为符合的内容分配内存,省。效率提升 15% 左右。
- 能不用正则,就不用正则,在分析的时候仔细阅读手册「字符串函数」部分。有没有你漏掉的好用的函数?
例如:
strpbrk()
strncasecmp()
strpos()/strrpos()/stripos()/strripos()
加速 strtr
如果需要转换的全是单个字符的时候,用字符串而不是数组来做 strtr:
<?php
$addr = strtr($addr, "abcd", "efgh"); // good
$addr = strtr($addr, array('a' => 'e',
// ...
)); // bad
?>
效率提升:10 倍。
不要做无谓的替换
即使没有替换,str_replace 也会为其参数分配内存。很慢!解决办法:
- 用 strpos 先查找(非常快),看是否需要替换,如果需要,再替换
效率:
- 如果需要替换:效率几乎相等,差别在 0.1% 左右。
- 如果不需要替换:用 strpos 快 200%。
邪恶的 @ 操作符
不要滥用 @ 操作符。虽然 @ 看上去很简单,但是实际上后台有很多操作。用 @ 比起不用 @,效率差距:3 倍。
特别不要在循环中使用 @,在 5 次循环的测试中,即使是先用 error_reporting(0) 关掉错误,在循环完成后再打开,都比用 @ 快。
善用 strncmp
当需要对比「前n个字符」是否一样的时候,用 strncmp/strncasecmp,而不是 substr/strtolower,更不是PCRE,更千万别提ereg。strncmp/strncasecmp 效率最高(虽然高得不多)。
慎用substr_compare(PHP5 ONLY)
按照上面的道理,substr_compare应该比先substr再比较快咯。答案是否定的,除非:
- 无视大小写的比较
- 比较较大的字符串
不要用常量代替字符串
为什么:
- 需要查询杂凑表两次
- 需要把常量名转换为小写(进行第二次查询的时候)
- 生成 E_NOTICE 警告
- 会建立临时字符串
效率差别:700%。
不要把 count/strlen/sizeof 放到 for 循环的条件语句中
Tips: 我的个人做法
<?php
for ($i = 0, $max = count($array);$i < $max; ++$i);
?>
效率提升相对于:
- count 50%
- strlen 75%
短的代码不一定快
<?php
// longest
if ($a == $b) {
$str .= $a;
} else {
$str .= $b;
}
// longer
if ($a == $b) {
$str .= $a;
}
$str .= $b;
// short
$str .= ($a == $b ? $a : $b);
?>
你觉得哪个快?
效率比较:
- longest: 4.27
- longer: 4.43
- short: 4.76
不可思议?再来一个:
<?php
// original
$d = dir('.');
while (($entry = $d->read()) !== false) {
if ($entry == '.' || $entry == '..') {
continue;
}
}
// versus
glob('./*');
// versus (include . and ..)
scandir('.');
?>
哪个快?
效率比较:
- original: 3.37
- glob: 6.28
- scandir: 3.42
- original without OO: 3.14
- SPL (PHP5): 3.95
画外音:从此也可以看出来 PHP5 的面向对象效率提高了很多,效率已经和纯函数差得不太多了。
提高 PHP 文件访问效率
需要包含其他 PHP 文件的时候,使用完整路径,或者容易转换的相对路径。
<?php
include 'file.php'; // bad approach
incldue './file.php'; // good
include '/path/to/file.php'; // ideal
?>
物尽其用
PHP 有很多扩展和函数可用,在实现一个功能的之前,应该看看 PHP 是否有了这个功能?是否有更简单的实现?
<?php
$filename = "./somepic.gif";
$handle = fopen($filename, "rb");
$contents = fread($handle, filesize($filename));
fclose($handle);
// vs. much simpler
file_get_contents('./somepic.gif');
?>
关于引用的技巧
引用可以:
- 简化对复杂结构数据的访问
- 优化内存使用
<?php
$a['b']['c'] = array();
// slow 2 extra hash lookups per access
for ($i = 0; $i < 5; ++$i)
$a['b']['c'][$i] = $i;
// much faster reference based approach
$ref =& $a['b']['c'];
for ($i = 0; $i < 5; ++$i)
$ref[$i] = $i;
?>
<?php
$a = 'large string';
// memory intensive approach
function a($str){
return $str.'something';
}
// more efficient solution
function a(&$str){
$str .= 'something';
}
?>
==============================================
参考资料
http://ilia.ws
Ilia 的个人网站,Blog,他参与的开发以及出版的一些稿物链接等等。
http://ez.no
eZ components 官方网站,eZ comp 是针对 PHP5 的开源通用库,以效率为己任,Ilia 也参与了开发。
http://phparch.com
php|architect,不错的 php 出版商/培训组织。买不起或者买不到的话,网上可以下到很多经典的盗版。
http://talks.php.net
PHP 会议上的演讲合集,现在还不是很丰富,不过内容都是让人一看就容易废寝忘食的好东东,推荐早上睡眼朦胧的时候或者吃完午饭仔细研究,否则你会忘记吃饭和睡觉的!