Наш чат в Telegram для обмена идеями, проектами, мыслями, людьми в сфере ИТ г.Ростова-на-Дону: @it_rostov

Ускорение PHP-программ

Как без особых усилий заставить PHP-код работать на порядок быстрее ? Перед тем как задаваться вопросами кеширования и масштабирования стоит попробовать оптимизировать код. Есть ряд несложных правил:

  • Выносите функции и вычисления из циклов. Такое:
    $ar=array(1,2,2,3,4,4);
    for($i=0;$i<count($ar);$i++){
        $a=12342+sqrt($b)/12123+$i;
    }
    Замените на
    $ar=array(1,2,2,3,4,4);
    $e=12342+sqrt($b)/12123;
    $c=count($ar);
    for($i=0;$i<$c;$i++){
        $a=$e+$i;
    }
    Альтернативный вариант:
    for($i=count($ar)-1;$i>=0;$i--){
    
    }
  • Выносите переменные из строк. Примерно так: <?php $x="test=".$test; ?> , это на 25-40% быстрее.
  • Имена переменных должны быть короче 7 символов. Это быстрее на 15%.
  • В массивах не стоит обращаться к элементам по имени и без кавычек. Особенно в многомерных. Сэкономите 40%.
  • А если обращение к элементу массива вынести из строки (как в пункте 1), то можно сэкономить ещё 25-30%.
  • Используйте Perl-ориентированные регулярные выражения. Или хотя бы сравнивайте их с POSIX-аналогами по скорости выполнения. Скорость может отличаться до 200% на разных запросах.
  • SizeOf() быстрее Count(), а в циклах sizeof лучше вообще заменить на переменную, for и while практически не отличимы, для перебора простых индексных массивов нужно использовать for или while, а для ассоциативных массивов - foreach. Плюс 30% при соблюдении.
  • Для чтения файла file() быстрее, чем fopen()+цикл. Ускорение до 40%.

Прочитать про всё это подробнее, посмотреть графики скорости и загрузить все примеры можно вот тут: "Оптимизация программ на PHP".

Еще про оптимизацию....

При вставке кусков PHP-кода в HTML страницы всегда используйте полные открывающие и закрывающие скобки <?php и ?>! Это обезопасит Вас от вариаций настройки php.ini short_open_tag на разных серверах и возможно сэкономит много времени при переносе или загрузке проектов на разные сервера.

Старайтесь использовать функцию вывода echo вместо printf и sprintf там где возможно. Нет надобности использовать эти функции , так как они выполняются медленней потому, что созданы для интерпретации и вывода строки с ее обработкой, подстановкой значений, в отформатированном виде. О чем и говорит буква f в конце названия этих 2-х функций.

Плохо:

sprintf('мама');
printf('папа');

Хорошо:

echo 'мама';
echo 'папа';

По тем же причинам используйте одинарные кавычки там где это возможно и пользуйтесь оператором "." для склейки строк, вместо прямой подстановки переменный в строку, заключенную в кавычки.

Лучший вариант (самый быстрый)

echo 'Вес равен: '.$weight;

Худший вариант (медленный):

echo "Вес равен: $weight";

Если Вам нужно проверить не равно ли возвращенное значение функции нулю (а функция сама по себе возвращает только положительные или только отрицательные значения), то лучше использовать оператор сравнения. Он выполняется быстрей, нежели конкретное сравнение значений.

Плохо:

$i = 0;
if ($i != 0)
 {
     //Не равно
 }
else
 {
     //Равно
 }

Хорошо:

$i = 0;
if ($i > 0)
 {
     //Не равно
 }
 else
 {
     //Равно
 }

Нужно также учитывать, что если строка принимает только пустое значения, либо пользовательские строковые данные, то вместо сравнения строки со строкой, для выявления ее пустоты, так же можно использовать сравнение с нулем, которые выполнится быстрее.

Для проверки строки на пустоту используйте функцию trim($str); Она не только проверит заполнена ли строка, но также обрежет несущественные символы - пробелы (табуляции, white-spaces) и вернет положительное значение, в случае если в строке ей действительно какие то значимые символы.

Плохо:

if ($str != '')
 {
     //обработка строки
 }

Надо:

if (trim($str))
 {
     //обработка строки
 }

Для получения данных из форм методом Get и Post лучше использовать следующий минимальный набор самописных функций:

GetParam ($array, $value, $default = '')
 {
     return (isset($array[$value])) ? $array[$value] : $default;
 }

GetParamSafe ($array, $value, $default = '')
 {
     return (isset($array[$value])) ? addslashes($array[$value]) : $default;
 }

Функция GetParam($_POST, 'myvar', 'empty') к примеру коректно получит данные из $_POST['myvar'], и в случае если $_POST переменная не существует вернет значение по умолчанию, без всяких Waring и Notice. Фунция GetParamSafe($_POST, 'myvar', 'empty') делает ту же операцию, только возвращает экранированную переменную. Для защиты от SQL инъекций к примеру. А данная конструкция позволяет получить целочисленное число из $_POST.

intval(GetParam($_POST, 'myvar', 'empty')):

В случае если в массиве $_POST лежало совсем не число функия вернет 0;

Для простого сравнения строк не используйте preg_match() или preg_match_all(). Используйте strstr() и strpos().

При получении строк из базы данных (MySQL к примеру) старайтесь использовать функцию mysql_fetch_object. К примеру при изменении кода запроса с

$query = "SELECT field7, field3 FROM mytable WHERE id = 5"

на

$query = "SELECT * FROM mytable WHERE id = 5"

код вывода строки полученной из этих запросов

$row = mysql_fetch_array(mysql_query($query));
echo $row[0].'-->'.$row[1]; //перестанет работать, в то время, как

$row = mysql_fetch_object(mysql_query($query));
echo $row->field7.'-->'.$row->field3; // останется работоспособным.

При использовании сессий для авторизации на сайте, храните в сессии хотя бы IP-адрес, с которого был совершен вход. Так же проверяйте IP входа с текущим IP адресом каждый раз при выполнении закрытого скрипта. Например если злоумышленнику удастся украсть название сессии, то войти он в закрытую часть уже не сможет. Потому что в общем случае у него будет другой IP-адрес.

При формировании больших запросов вставки данных в БД через insert все строчки старайтесь поместить в один-три insert'а. Выполнение каждой строчки отдельно не только загрузит сервер БД, но и задержит работу Вашего скрипта.

В случае если необходимо в разных местах (разных классах) одной системы использовать одни и те же сложно вычисляемые данные (например которые достаются из БД через запрос с последющей обработкой строк), старайтесь их вычислять единожды, хранить глобально для всей системы и передавать в класс(скрипт) один раз, непосредственно при создании класса (подключении скрипта)

При больших нагрузках на Web-сервер задумайтесь над использованием стандартных решений для включения кэша(кэш-технологии). Например, бесплатный PHP класс JCache_Lite_Function.

При проектировании/разработке больших систем отдавайте предпочтение Объектно-Ориентированному программированию с использование шаблонов проектирования. Наиболее частые шаблоны: MVC, PageController, BodyHandler, Fabric...

Читать дальше: Примеры скриптов на PHP


.