Loading... # [PHP性能优化技巧【译】](https://blog.p2hp.com/archives/597) 今天在逛微博的时候看到了别人转的一篇谷歌的的文章《PHP performance tips》,看了觉得很有用,于是就翻译保存下来。 概括出你的代码的瓶颈所在 当你尝试让你的网站更快的时候,你需要记住Hoare的名言:不成熟的优化是万恶之源。在你修改你的代码之前,你需要明确是什么造成了网站运行的慢。 更新你的PHP版本 这些年,PHP开发团队为PHP引擎做了很多重大的性能的提升。如果你的服务器上还是运行的比较老的PHP版本(比如PHP3 或者 PHP4)的话,那在你优化你的代码之前你需要先升级你的PHP版本 * Migrating from PHP 4 to PHP 5.0.x * Migrating from PHP 5.0.x to PHP 5.1.x * Migrating from PHP 5.1.x to PHP 5.2.x 使用缓存 使用类似Memcache或者Smarty这样支持缓存的模板引擎的缓存模块,通过缓存数据库中数据或者已经渲染好的页面的方法可以提高网站的性能。 使用输出缓冲 PHP使用内存缓冲区来保存脚本要输出的所有的数据。由于缓存区必须被填满之后才会将数据发送给用户,所以你的页面看上去会比较慢 * Output Buffering Control 尽量避免使用setter和getter 当使用PHP的类的时候,直接使用类中的属性比使用setter和getter可以节省编码时间,也能让你的脚本的运行速度更快。 在下面的例子中,dog类中使用setName()和getName()方法来访问name属性: class dog { public $name = ''; public function setName($name) { $this->name = $name; } public function getName() { return $this->name; } } 注意,在这里setName和getName除了存储和返回name的值,其他什么也不做。 $rover = new dog(); $rover->setName('rover'); echo $rover->getName(); 直接设置和调用name属性可以让运行效率提升一倍,也能减少我们的开发时间。 $rover = new dog(); $rover->name = 'rover'; echo $rover->name; 不要随便就复制变量 有时候为了使 PHP 代码更加整洁,一些 PHP 新手(包括我)会把预定义好的变量复制到一个名字更简短的变量中,其实这样做的结果是增加了一倍的内存消耗,只会使程序更加慢。试想一下,在 下面的例子中,如果用户恶意插入 512KB 字节的文字到文本输入框中,这样就会导致 1MB 的内存被消耗! BAD: $description = $_POST['description']; echo $description; GOOD: echo $_POST['description']; 对字符串使用单引号 PHP 引擎允许使用单引号和双引号来封装字符串变量,但是这个是有很大的差别的!使用双引号的字符串告诉 PHP 引擎首先去读取字符串内容,查找其中的变量,并改为变量对应的值。一般来说字 符串是没有变量的,所以使用双引号会导致性能不佳。最好是使用字符串连接而不是双引号字符串。 BAD: $output = "This is a plain string"; GOOD: $output = 'This is a plain string'; GOOD经过测试并不是原文上所说的那样,所以加以改正: $type = "mixed"; $output = "This is a $type string"; BAD: $type = 'mixed'; $output = 'This is a ' . $type .' string'; 使用 echo 函数来输出字符串 使用 echo() 函数来打印结果出了有更容易阅读之外,在下个例子中,你还可以看到有更好的性能。 BAD: print($myVariable); GOOD: echo $myVariable; 不要在 echo 中使用连接符 很多 PHP 程序员(有包括我)不知道在用 echo 输出多个变量的时候,其实可以使用逗号来分开的,而不必用字符串先把他们先连起来,如下面的第一个例子中,由于使用了连接符就会有性能问题,因为这样就会需要 PHP 引擎首先把所有的变量连接起来,然后在输出,而在第二个例子中,PHP 引擎就会按照循序输出他们。 BAD: echo 'Hello, my name is' . $firstName . $lastName . ' and I live in ' . $city; GOOD: echo 'Hello, my name is' , $firstName , $lastName , ' and I live in ' , $city; 使用 switch/case 代替 if/else 对于只有单个变量的判断,使用 switch/case 语句而不是 if/else 语句,会有更好的性能,并且代码更加容易阅读和维护。 BAD: if($_POST['action'] == 'add') { addUser(); } elseif ($_POST['action'] == 'delete') { deleteUser(); } elseif ($_POST['action'] == 'edit') { editUser(); } else { defaultAction(); } GOOD: switch($_POST['action']) { case 'add': addUser(); break; case 'delete': deleteUser(); break; case 'edit': editUser(); break; default: defaultAction(); break; } 避免在循环里面执行sql 一个很常见的问题就是在循环里面执行sql。这样多次连接并从数据库得到数据会显著的将脚本运行速度变慢。在下面的例子中,你可以改变之前的循环执行,将SQL改造成一条单独的sql然后一次性插入你的所有数据。 foreach ($userList as $user) { $query = 'INSERT INTO users (first_name,last_name) VALUES("' . $user['first_name'] . '", "' . $user['last_name'] . '")'; mysql_query($query); } 插入语句: INSERT INTO users (first_name,last_name) VALUES("John", "Doe") 代替使用循环,可以将所有的数据组合进一条单独的sql中并执行。 $userData = array(); foreach ($userList as $user) { $userData[] = '("' . $user['first_name'] . '", "' . $user['last_name'] . '")'; } $query = 'INSERT INTO users (first_name,last_name) VALUES' . implode(',', $userData); mysql_query($query); 插入语句: INSERT INTO users (first_name,last_name) VALUES("John", "Doe"),("Jane", "Doe")... 自己第一次翻译,英语比较差,所以练习一下,也增进自己的自己,好记性不如烂笔头嘛。 最后修改:2023 年 08 月 07 日 © 允许规范转载 赞 如果觉得我的文章对你有用,请随意赞赏