分类目录归档:php

主要更新PHP相关的文章

json在php中的函数实例

<?php

$a=array(‘user’,’pwd’);
//加密 对变量进行 JSON 编码
$jiami=json_encode($a);

//解密 对 JSON 格式的字符串进行解码,转换为 PHP 变量

$jiemi=json_decode($jiami);   //不加true 返回的是对象

print_r($jiami);
$jiemi=json_decode($jiami,true);  //加true 返回的是数组
print_r($jiami);

echo “<br />”;

json也可以转换类

class jso{
public $name=’haha’;
protected $ptname=’xiaomi’;
private $pname=’mimi’;
public function getName(){
return $this->name;
}
}

$soobj = new jso();
$ac=json_encode($soobj);

//echo $soobj->getName();
print_r($ac);
echo “<br />”;
$jsonstr='{“key”:”value”,”key1″:”value1″}’;
$ad=json_decode($jsonstr);
//$ad=json_decode($jsonstr,true);
print_r($ad);

$json = ‘{“a”:1,”b”:2,”c”:3,”d”:4,”e”:5}’;
$a=json_encode($json,true);
var_dump($a);

PHP性能优化大全

PHP优化对于PHP的优化主要是对php.ini中的相关主要参数进行合理调整和设置,以下我们就来看看php.ini中的一些对性能影响较大的参数应该如何设置。

# vi /etc/php.ini

(1) PHP函数禁用找到:

disable_functions =
该选项可以设置哪些PHP函数是禁止使用的,PHP中有一些函数的风险性还是相当大的,可以直接执行一些系统级脚本命令,如果允许这些函数执行,当PHP程序出现漏洞时,损失是非常严重的!以下我们给出推荐的禁用函数设置:

disable_functions = phpinfo,passthru,exec,system,popen,chroot,escapeshellcmd,escapeshellarg,shell_exec,proc_open,proc_get_status

需注意:如果您的服务器中含有一些系统状态检测的PHP程序,则不要禁用shell_exec,proc_open,proc_get_status等函数。

(2) PHP脚本执行时间找到:

max_execution_time = 30

该选项设定PHP程序的最大执行时间,如果一个PHP脚本被请求,且该PHP脚本在max_execution_time时间内没能执行完毕,则PHP不再继续执行,直接给客户端返回超时错误。没有特殊需要该选项可保持默认设置30秒,如果您的PHP脚本确实需要长执行时间则可以适当增大该时间设置。

(3) PHP脚本处理内存占用找到:

memory_limit = 8M

该选项指定PHP脚本处理所能占用的最大内存,默认为8MB,如果您的服务器内存为1GB以上,则该选项可以设置为12MB以获得更快的PHP脚本处理效率。

(4) PHP全局函数声明找到:

register_globals = Off

网络上很多关于PHP设置的文章都推荐将该选项设置为On,其实这是一种及其危险的设置方法,很可能引起严重的安全性问题。如果没有特殊的需要,强烈推荐保留默认设置!

(5) PHP上传文件大小限制找到:

upload_max_filesize = 2M

该选项设定PHP所能允许最大上传文件大小,默认为2MB。根据实际应用需求,可以适当增大该设置。

(6) Session存储介质找到:

session.save_path

 

如果你的PHP程序使用Session对话,则可以将Session存储位置设置为/dev/shm,/dev/shm是Linux系统独有的TMPFS 文件系统,是以内存为主要存储方式的文件系统,比RAMDISK更优秀,因为可以使用DISKSWAP作为补充,而且是系统自带的功能模块,不需要另行配置。想想看,从磁盘IO操作到内存操作,速度会快多少?只是需要注意,存储在/dev/shm的数据,在服务器重启后会全部丢失。不过这对于 Session来说是无足轻重的。

 

 

由于水平有限,有些还是不太明白为什么。如果有更好建议的欢迎随时补充!

0、用单引号代替双引号来包含字符串,这样做会更快一些。因为PHP会在双引号包围的字符串中搜寻变量,单引号则不会,注意:只有echo能这么做,它是一种可以把多个字符串当作参数的“函数”(译注:PHP手册中说echo是语言结构,不是真正的函数,故把函数加上了双引号)。
PS:在单引号中,PHP不会自动搜寻变量、转义字符等,因此效率上快很多。而一般来说字符串是没有变量的,所以使用双引号会导致性能不佳。

1、如果能将类的方法定义成static,就尽量定义成static,它的速度会提升将近4倍。
PS:事实上,function、method、static method的速度不会有太大差异。

2、$row[’id’] 的速度是$row[id]的7倍。
PS:不太懂,貌似差异只有后者会先判断id这个宏是否存在,如果不存在则自动转变为字符串。

3、echo 比 print 快,并且使用echo的多重参数(译注:指用逗号而不是句点)代替字符串连接,比如echo $str1,$str2。

PS:如果使用echo $str1.$str2 就会需要 PHP 引擎首先把所有的变量连接起来,然后在输出,而echo $str1,$str2,PHP 引擎就会按照循序输出他们

4、在执行for循环之前确定最大循环数,不要每循环一次都计算最大值,最好运用foreach代替。
PS:像count、strlen这样的操作其实是O(1)的,因此不会带来太多消耗,当然避免每次循环都计算是比较好的策略。最好用foreach代替for,这个效率更高,如果考虑到 foreach($array as $var)每次拷贝的消耗,可以使用foreach($array as &$var)这样的引用。

5、注销那些不用的变量尤其是大数组,以便释放内存。
PS:如果没有记错的话,unset($array)不会立刻释放内存,但随时释放是个好习惯。

6、尽量避免使用__get,__set,__autoload。

7、require_once()代价昂贵。
PS:require_once和include_once需要判重,因此效率上要低,但是5.2版本后效率问题已经基本解决。

8、include文件时尽量使用绝对路径,因为它避免了PHP去include_path里查找文件的速度,解析操作系统路径所需的时间会更少。
PS:支持,尽量少用iniset()来设置include_path。

9、如果你想知道脚本开始执行(译注:即服务器端收到客户端请求)的时刻,使用$_SERVER[‘REQUEST_TIME’]要好于time()。
PS:$_SERVER[‘REQUEST_TIME’]保存了发起该请求时刻的时间戳,而time()则返回当前时刻的Unix时间戳。

10、函数代替正则表达式完成相同功能。
PS:这种函数是指strtok、strstr、strpos、str_replace、substr、explode、implode等等。

11、str_replace函数比preg_replace函数快,但strtr函数的效率是str_replace函数的四倍。
PS:字符串操作比正则替换要快。

12、如果一个字符串替换函数,可接受数组或字符作为参数,并且参数长度不太长,那么可以考虑额外写一段替换代码,使得每次传递参数是一个字符,而不是只写一行代码接受数组作为查询和替换的参数。
PS:需要考虑到内置函数和用户自定义函数的开销差异,恐怕这种做法得不偿失。

13、使用选择分支语句(译注:即switch case)好于使用多个if,else if语句。
PS:php中switch支持数值和字符串变量,比C的switch要好用,建议使用。

14、用@屏蔽错误消息的做法非常低效,极其低效。
PS:有什么替代方法吗?没有的话还是不得不用的……

15、打开apache的mod_deflate模块,可以提高网页的浏览速度。

16、数据库连接当使用完毕时应关掉,不要用长连接。
PS:在连接之前,最好设置一下相应的超时机制,例如链接超时、读写超时、等待超时等。

17、错误消息代价昂贵。

18、在方法中递增局部变量,速度是最快的。几乎与在函数中调用局部变量的速度相当。

19、递增一个全局变量要比递增一个局部变量慢2倍。

20、递增一个对象属性(如:$this->prop++)要比递增一个局部变量慢3倍。

21、递增一个未预定义的局部变量要比递增一个预定义的局部变量慢9至10倍。

22、仅定义一个局部变量而没在函数中调用它,同样会减慢速度(其程度相当于递增一个局部变量)。PHP大概会检查看是否存在全局变量。

23、方法调用看来与类中定义的方法的数量无关,因为我(在测试方法之前和之后都)添加了10个方法,但性能上没有变化。

24、派生类中的方法运行起来要快于在基类中定义的同样的方法。

25、调用带有一个参数的空函数,其花费的时间相当于执行7至8次的局部变量递增操作。类似的方法调用所花费的时间接近于15次的局部变量递增操作。

26、Apache解析一个PHP脚本的时间要比解析一个静态HTML页面慢2至10倍。尽量多用静态HTML页面,少用脚本。

27、除非脚本可以缓存,否则每次调用时都会重新编译一次。引入一套PHP缓存机制通常可以提升25%至100%的性能,以免除编译开销。

28、尽量做缓存,可使用memcached。memcached是一款高性能的内存对象缓存系统,可用来加速动态Web应用程序,减轻数据库负载。对运算码 (OP code)的缓存很有用,使得脚本不必为每个请求做重新编译。

29、当操作字符串并需要检验其长度是否满足某种要求时,你想当然地会使用strlen()函数。此函数执行起来相当快,因为它不做任何计算,只返回在zval 结构(C的内置数据结构,用于存储PHP变量)中存储的已知字符串长度。但是,由于strlen()是函数,多多少少会有些慢,因为函数调用会经过诸多步骤,如字母小写化(译注:指函数名小写化,PHP不区分函数名大小写)、哈希查找,会跟随被调用的函数一起执行。在某些情况下,你可以使用isset() 技巧加速执行你的代码。

(举例如下)

if (strlen($foo) < 5) { echo “Foo is too short”$$ }

(与下面的技巧做比较)

if (!isset($foo{5})) { echo “Foo is too short”$$ }

调用isset()恰巧比strlen()快,因为与后者不同的是,isset()作为一种语言结构,意味着它的执行不需要函数查找和字母小写化。也就是说,实际上在检验字符串长度的顶层代码中你没有花太多开销。
PS:长见识了。

30、当执行变量$i的递增或递减时,$i++会比++$i慢一些。这种差异是PHP特有的,并不适用于其他语言,所以请不要修改你的C或Java代码并指望它们能立即变快,没用的。++$i更快是因为它只需要3条指令(opcodes),$i++则需要4条指令。后置递增实际上会产生一个临时变量,这个临时变量随后被递增。而前置递增直接在原值上递增。这是最优化处理的一种,正如Zend的PHP优化器所作的那样。牢记这个优化处理不失为一个好主意,因为并不是所有的指令优化器都会做同样的优化处理,并且存在大量没有装配指令优化器的互联网服务提供商(ISPs)和服务器。

31、并不是事必面向对象(OOP),面向对象往往开销很大,每个方法和对象调用都会消耗很多内存。

32、并非要用类实现所有的数据结构,数组也很有用。

33、不要把方法细分得过多,仔细想想你真正打算重用的是哪些代码?

34、当你需要时,你总能把代码分解成方法。
PS:分解成方法要适当,行数少使用频率高的方法尽量用直接写代码,可以减少函数堆栈开销;且方法嵌套不宜过深,否则大大影响PHP的运行效率。

35、尽量采用大量的PHP内置函数。

36、如果在代码中存在大量耗时的函数,你可以考虑用C扩展的方式实现它们。

37、评估检验(profile)你的代码。检验器会告诉你,代码的哪些部分消耗了多少时间。Xdebug调试器包含了检验程序,评估检验总体上可以显示出代码的瓶颈。

38、mod_zip可作为Apache模块,用来即时压缩你的数据,并可让数据传输量降低80%。

39、在可以用 file_get_contents替代file、fopen、feof、fgets等系列方法的情况下,尽量用file_get_contents,因为他的效率高得多!但是要注意file_get_contents在打开一个URL文件时候的PHP版本问题;
PS:这个要记住,尽量使用file_get_contents和file_put_contents,不需要自己判断文件句柄打开是否成功。

40、尽量的少进行文件操作,虽然PHP的文件操作效率也不低的;

41、优化Select SQL语句,在可能的情况下尽量少的进行Insert、Update操作(在update上,我被恶批过);

42、尽可能的使用PHP内部函数(但是我却为了找个PHP里面不存在的函数,浪费了本可以写出一个自定义函数的时间,经验问题啊!);
PS:内置函数比用户自定义函数效率高了将近一个数量级。

43、循环内部不要声明变量,尤其是大变量:对象(这好像不只是PHP里面要注意的问题吧?);
PS:这个必须的,变量过多或者过大时,每次重分配的开销就无法忽略。

44、多维数组尽量不要循环嵌套赋值;

45、在可以用PHP内部字符串操作函数的情况下,不要用正则表达式;

46、foreach效率更高,尽量用foreach代替while和for循环;

47、用单引号替代双引号引用字符串;
PS:晕,这个不就是第一条吗?

48、“用i+=1代替i=i+1。符合c/c++的习惯,效率还高”;

49、对global变量,应该用完就unset()掉;

PHP写的发送邮件(测试无问题)

<?php
function send_mail($to, $subject = ‘No subject’, $body) {
$loc_host = “test”;                 //发信计算机名,可随意
$smtp_acc = “xxx@qq.com”;        //Smtp认证的用户名,类似fuweng@im286.com,或者fuweng
$smtp_pass=”*******”;              //Smtp认证的密码,一般等同pop3密码
$smtp_host=”smtp.qq.com”;    //SMTP服务器地址,类似 smtp.tom.com
$from=”xxxx@qq.com”;              //发信人Email地址,你的发信信箱地址
$headers = “Content-Type: text/plain; charset=”utf-8″
Content-Transfer-Encoding: base64″;
$lb=”
“;                         //linebreak

$hdr = explode($lb,$headers);     //解析后的hdr
if($body) {
$bdy = preg_replace(“/^./”,”..”,explode($lb,$body));   //解析后的Body
}

$smtp = array(
//1、EHLO(是客户端向对方邮件服务器发出的标识自己身份的命令),期待返回220或者250
array(“EHLO “.$loc_host.$lb,”220,250”,”HELO error: “),
//2、发送Auth Login,期待返回334
array(“AUTH LOGIN”.$lb,”334″,”AUTH error:”),
//3、发送经过Base64编码的用户名,期待返回334
array(base64_encode($smtp_acc).$lb,”334″,”AUTHENTIFICATION error : “),
//4、发送经过Base64编码的密码,期待返回235
array(base64_encode($smtp_pass).$lb,”235”,”AUTHENTIFICATION error : “)
);
//5、发送Mail From,期待返回250
$smtp[] = array(“MAIL FROM: <“.$from.”>”.$lb,”250″,”MAIL FROM error: “);
//6、发送Rcpt To。期待返回250(0K)
$smtp[] = array(“RCPT TO: <“.$to.”>”.$lb,”250″,”RCPT TO error: “);
//7、发送DATA,期待返回354
$smtp[] = array(“DATA”.$lb,”354″,”DATA error: “);
//8.0、发送From
$smtp[] = array(“From: “.$from.$lb,””,””);
//8.2、发送To
$smtp[] = array(“To: “.$to.$lb,””,””);
//8.1、发送标题
$smtp[] = array(“Subject: “.$subject.$lb,””,””);
//8.3、发送其他Header内容
foreach($hdr as $h) {$smtp[] = array($h.$lb,””,””);}
//8.4、发送一个空行,结束Header发送
$smtp[] = array($lb,””,””);
//8.5、发送信件主体
if($bdy) {foreach($bdy as $b) {$smtp[] = array(base64_encode($b.$lb).$lb,””,””);}}
//9、发送“.”表示信件结束,期待返回250
$smtp[] = array(“.”.$lb,”250″,”DATA(end)error: “);
//10、发送Quit,退出,期待返回221
$smtp[] = array(“QUIT”.$lb,”221″,”QUIT error: “);

//打开smtp服务器端口
$fp = @fsockopen($smtp_host, 25);
if (!$fp) echo “Error: Cannot conect to “.$smtp_host.””;
while($result = @fgets($fp, 1024)){
if(substr($result,3,1) == ” “) { break; }
}

$result_str=””;
//发送smtp数组中的命令/数据
foreach($smtp as $req){
//发送信息
@fputs($fp, $req[0]);// fputs(字符串,文件指针)
//如果需要接收服务器返回信息,则
if($req[1]){
//接收信息
while($result = @fgets($fp, 1024)){
if(substr($result,3,1) == ” “) { break; }
};
if (!strstr($req[1],substr($result,0,3))){
$result_str.=$req[2].$result.””;
}
}
}
//关闭连接
@fclose($fp);
return $result_str;
}
if(@$_POST[‘sub’])
{
$email_from=@$_POST[’email_from’];
$email_title=@$_POST[’email_title’];
$email_content=@$_POST[’email_content’];

echo  send_mail($email_from,$email_title,$email_content);
}
?>
<!DOCTYPE HTML PUBLIC “-//W3C//DTD HTML 4.01 Transitional//ENhttp://www.w3.org/TR/html4/loose.dtd”>
<html lang=”en”>
<head>
<meta http-equiv=”Content-Type” content=”text/html;charset=UTF-8″>
<title></title>
</head>
<body>
<form method=”post”>
收件人邮箱<input type=”text” name=”email_from” onkeyup=”this.value.replace(^[w-]+(.[w-]+)*@[w-]+(.[w-]+)+$)”>
标题<input type=”text” name=”email_title”>
内容<input type=”text” name=”email_content”>
<input type=”submit” name=”sub” value=”发送右键”>
</form>
</body>
</html>

wamp配置多站点虚拟目录

首先在httpd.conf里面找到
LoadModule vhost_alias_module modules/mod_vhost_alias.so去掉#
Include conf/extra/httpd-vhosts.conf去掉#
<Directory />
Options FollowSymLinks
AllowOverride  None
Order  deny,allow
Deny from all
</Directory>

<Directory />
Options Indexes FollowSymLinks
AllowOverride None
</Directory>
在httpd-vhosts.conf加
<VirtualHost *:80>
DocumentRoot D:/z
ServerName www.xiaobo.com
<Directory “D:/z”>
Options Indexes FollowSymLinks
AllowOverride None
Order allow,deny
Allow from all
</Directory>
</VirtualHost>

PHP写无限分类菜单

/**

* find_child,对于数组$arr,给定一个id,获取其子节点构成的一维数组

* @param $arr array 要遍历的数组

* @param $id int 该条记录的id,默认值为0

* @return array 返回$id记录对应的子节点数组

*/

function find_child($arr,$id=0){

$child = array();

foreach ($arr as $v) {

if($v[‘pid’] == $id){

$child[] = $v;

}

}

return $child;

}

/**

* get_tree,对于数组$arr,给定一个根,获取其子孙节点构成的多维数组

* @param $arr array 要遍历的数组

* @param $root_id int 根节点 id,默认为0

* @return array 返回$root_id节点对应的子孙节点构成的多维数组

*/

function get_tree($arr,$root_id=0){

$child = find_child($arr,$root_id);

if(empty($child)){

return null;

}

foreach ($child as $k => $v) {

$current_tree = get_tree($arr,$v[‘id’]);

if ($current_tree != null) {

$child[$k][‘child’] = $current_tree;

}

}

return $child;

}

PHP二分查找(必须掌握)

<?php
/*
*  二分查找(必须是有顺序数)必须掌握
*  @arr 要被查找的数组
*  @findvl  要查找的数
*  @right   右边的数组
*  @left    左边数组
*  @middle  中间的数
*/
$arr=array(-1,0,40,90);
function binarysearch(&$arr,$findval,$left,$right)
{
if($right<$left)
{
echo “找不到此数”;
return ;
}
$middle=round(($right+$left)/2);
if($findval>$arr[$middle])
{
binarysearch($arr,$findval,$middle+1,$right);
}else if($findval<$arr[$middle])
{
binarysearch($arr,$findval,$left,$middle-1);
}else
{
echo $middle;
}
}
binarysearch($arr,0,0,count($arr));