Плюсы использования шаблонов:
• Предельно просто и ясно
• Для изменения шаблонов не требуется знание PHP
Минусы использования шаблонов:
• Более медленный способ - ведь надо сканировать весь шаблон и лишь потом
выводить данные
• Сложнее внедрить на практике
15. Использование устаревшего синтаксиса и функций
Некоторые программисты вновь и вновь используют старые библиотеки и старые
наработки. Например, код, написанный еще под PHP 2, до сих пор используется с
PHP4, хотя уже начиная с версии PHP3 были добавлены стандартные функции,
реализующие то же самое.
Использование устаревших функций и синтаксиса могут снизить скорость выполнения
кода и, к тому же, сделать его нечитаемым. Другие программисты могут быть
незнакомы со старыми функциями. Но тем не менее, если вы встретили участок
старого кода, не обязательно его переписывать с учетом новых правил языка.
Просто не надо его использовать при написании новых программ.
Пример использования старых языковых конструкций:
<?php
// Старый стиль
while (1):
print "5";
if ( $idx++ == 5 ):
break;
endif;
endwhile;
// Лучше написать так
// (впрочем, код можно оптимизировать)
while (1)
{
print "5";
if ( $idx++ == 5 ) {
break;
}
}
?>
Почему же следует следовать новым стандартам? Причины следующие:
• Использование старых конструкций не очень распространено и, таким образом,
новички в PHP будут в замешательстве, увидев два разных варианта синтаксиса.
• Старый синтаксис отличается от синтаксиса других языков программирования, и,
следовательно, при переходе с другого языка на PHP программисту будет сложнее
понять и привыкнуть.
• Но самое главное - в одной из новых версий, возможно, будет исключена
поддержка старого синтаксиса, тем самым это заставит вас переписать код заново.
Как бы то ни было, скобки всегда останутся частью языка PHP.
Подобные участки кода можно встретить во многих программах. Вам, как правило,
следует руководствоваться правилами, приведенными в документации по PHP, большей
частью обновленной - в ней отражается развитие языка. Периодически
просматривайте документацию, ведь язык развивается, добавляются новые функции.
Таким образом, вам никогда не придется писать пользовательские функции,
выполняющие ту же работу, что и стандартные.
Резюме
В этой статье мы рассмотрели первые 7 из 21 наиболее общих ошибок PHP
программиста. Как правило, они не нарушают работоспособности программ, но, тем
не менее, их следует избегать:
• Необоснованное применение функции printf(): Ее следует использовать только для
вывода форматированных данных.
• Неправильное применение семантики языка: Многие программисты не имеют
достаточно времени, чтобы разобраться во всех тонкостях языка, что впоследствии
выражается в ошибочном коде.
• Плохо комментированный код: Всегда пишите комментарии! Перед каждой функцией
указывайте, что делает данная функция, и какие аргументы она требует. Также
комментируйте сложные участки кода и внесенные изменения.
• Слишком много временных переменных: Временные переменные хорошо использовать
для предотвращения повторного вызова функций или последовательностей функций.
• Изобретаем велосипед - переписываем стандартную функцию: Сначала загляните в
руководство по PHP - не описана ли там функция, которую вы собираетесь написать
для, казалось бы, расширения набора стандартных функций PHP.
• Смешан PHP и HTML код: Попробуйте сделать код как можно более модульным. Потом
вам (и другим тоже) можно будет сменить дизайн страницы без изменения кода PHP.
• Используются старые языковые конструкции и устаревшие функции: То, что вы
можете сделать, не всегда следует делать. Загляните в документацию и литературу
по PHP как писать правильно. Отличные книги - "Разработка веб-приложений с
использованием PHP (Web Application Development with PHP)" и "Профессиональный
программист PHP (Professional PHP)". (Эх, где бы их еще найти! ;)) - прим.
переводчика)
14-7
Введение
Одна из наиболее сильных сторон PHP является, одновременно, и его слабой
стороной: PHP очень прост в изучении. Это привлекает многих людей; однако,
несмотря на его кажущуюся простоту, не так-то просто научиться использовать этот
язык правильно и эффективно.
Как правило, дело в недостаточной практике программирования. Неопытные
программисты становятся перед лицом необходимости создания сложных
веб-приложений. Поэтому сплошь и рядом допускаются ошибки, которых избежал бы
опытный программист, такие как необоснованное использование функции printf()или
неправильное использование семантики PHP.
В этой серии из трех статей представлены наиболее, по нашему мнению, характерные
ошибки. Эти ошибки можно классифицировать по нескольким категориям, от
"некритических" до "смертельных". Наряду с анализом этих ошибок, представлены
способы их избежания, а также некоторые "маленькие хитрости", накопленные за
многие годы практики программирования.
Часть 1: Описываются 7 "детских" ошибок (#21-15, в обратном порядке, в
соответствии со степенью серьёзности по нашей классификации). Такие ошибки не
вызывают серьёзных проблем, но приводят к уменьшению эффективности работы
программы, а также выражаются в громоздком трудночитаемом коде, в который, к
тому же, трудно вносить изменения.
Часть 2: Следующие 7 ошибок (#14-8) относятся к "серьёзным". Они ведут к ещё
более значительному уменьшению скорости выполнения кода, уменьшению безопасности
скриптов; код становится еще более запутанным.
Часть 3: Описания семи, последних, "смертельных" ошибок. Эти ошибки
концептуальны по своей природе и являются причиной появления ошибок, описанных в
1-ой и 2-ой частях статьи. Они включают и такие ошибки, как недостаточное
внимание, уделённое как проекту в целом, так и коду программы, в частности.
14. Пренебрежение правилами присвоения имён
Одна из наиболее серьёзных ошибок программиста - непродуманная система
именования переменных проекта. Нередко приходится тратить уйму времени на разбор
кода только потому, что автор вдруг решил ввести в программу переменные $fred и
$barney вместо ожидаемых $email и $name. Речь ведётся о реальном проекте, где не
менее реальный программист решил все переменные проекта назвать именами героев
мультсериала "Flinstones" (Это не шутка).
То как вы назовёте переменные и функции программы, определит во многом
читаемость её кода. Наиболее распространёнными ошибками являются имена:
• слишком короткие или наоборот, чрезмерно длинные;
• не связанные по смыслу с контекстом программы;
• не учитывающие регистрозависимость;
• замедляющие разбор и чтение кода (особенно это касается имён функций).
Именование переменных
Регистрозависимость
В PHP имена переменных регистрозависимы, то есть $user и $User - две записи в
списке переменных скрипта. Однако некоторые программисты активно пользуются этим
и производят на свет переменные с совершенно одинаковыми именами, но
использующими буквы разных регистров. Это отвратительная привычка. Регистр букв
никогда не должен быть единственным отличием двух переменных. Каждая переменная
на своём поле действия должна иметь уникальное имя.
Слишком короткие имена
Для обозначения переменных многие программисты используют одним им понятные
аббревиатуры. О чём впоследствии сильно жалеют, ибо смысл сокращения затерялся
во времени своего создания. Имя переменной должно отражать характер её значения,
то есть содержания и обозначаться полными словами или общепонятными
сокращениями.
Слишком длинные имена
С другой стороны, наблюдаются случаи злоупотребления длинными именами. Наиболее
общее правило: имя переменной должно состоять максимум из двух слов. Разделить
эти два слова мы можем, поставив understrike (то есть "_") или написав второе
слово с заглавной буквы.
Пример #1. Положительный.
Как правильно присваивать имена переменным:
foreach ($teachers as $teacher);
?>
Пример #2. Отрицательный.
Теперь рассмотрим несколько преувеличенные примеры того, как не следует
присваивать имена переменным:
$username_for_database = 'sterling';
$guMbi = 'secret'; // for the $password
foreach ($thelastnamesofteachers as
$TeaChER);
?>
Имена функций
Все правила, применяемые для имён переменных, годятся и для функций. Однако в
случае с функциями, грамматические реалии имеют большее значение.
Помните, что в PHP все функции, встроенные или определённые разработчиком, -
регистронезависимы.
Использование глаголов
Функции в PHP можно сравнить с какими-либо действиями, совершаемыми в реальном
мире. Таким образом, имена функций должны отражать эту направленность на
действие, то есть выражаться глаголами. Причём лучше в настоящем времени.
В качестве примера рассмотрим функцию, генерирующую Гауссовы случайные числа.
Предполагается, что из её имени мы должны понять, какая именно формула
используется в генерации числа. Вот так: generate_gaussian_rand().
Обратите внимание на использование глагола в имени функции. Именно глагол
помещает функцию в правильный контекст:
list ($num1, $num2) = generate_gaussian_rand();
list ($num3, $num4) = generate_gaussian_rand();
?>
Для сравнения, другой пример:
list ($num1, $num2) = gaussian_rand_generator();
list ($num1, $num2) = gaussian_rand_generator();
?>
Видите разницу? Во втором примере для обозначения действия использовано
существительное. И если назначение функции ещё прослеживается, название
затрудняет чтение кода.
Мораль: используйте глаголы!
13. Непродуманная работа с данными: бд и sql
Забавно иногда наблюдать, сколько разных уловок находят люди для организации
доступа к базам данных и получения выборки результатов. Среди прочих особенно
выделяются комбинации из веток if, циклов do..while, множественных запросов и
вызовов функции sql_result() внутри цикла for.
Чем, на их взгляд, они занимаются?
Код, основанный на методе научного тыка, говорит о недостаточно ясно
определённой организации работы с БД. Те, кто прилагают все свои усилия на
написание кода, а не на написание правильного кода, рискуют больше потерять, чем
заработать. Некорректная выборка данных - яркий тому пример. Некоторые
программисты не уделяют достаточно времени на тщательное продумывание этого
момента. Естественно, в реальной жизни может и не оказаться того "единственно
верного" способа выборки данных, но всегда найдётся тысяча "неверных", это
точно.
Ошибки в организации выборки данным можно разделить на три класса:
• неправильное использование функций обращения к БД
• ошибки SQL: запрашивается не то, что нужно
• обработка результатов выборки средствами PHP
Неправильное использование функций обращения к БД
Один из PHP-исходников предлагал следующий способ получения выборки из БД
(приведённый ниже код в проекте находится после сгенерированных SQL-запросов):
if (!($row = sql_fetch_row ($result))) {
print "Ошибка: не найдено ни одного ряда";
exit;
}
do {
print "$row[0]: $row[1]\n
\n";
}
while ($row = sql_fetch_row ($result));
?>
Примечание: в данном и последующих примерах $result является дескриптором
выборки или указателем на неё. Другими словами, был произведён запрос и получено
определённое множество рядов. Примеры демонстрируют методы эффективной обработки
этого множества.
В этом отрезке кода есть две ошибки:
• проверка на "ноль рядов" - это попытка получить хотя бы один.
• полученные данные не хранятся в ассоциативном массиве.
Проверка на "ноль рядов" ($result): неправильный подход
Задействовав функцию sql_fetch_row(), данный кусок кода предлагает косвенную
проверку выборки на наличие хотя бы одного ряда данных. Но ведь существует
прямой способ - это подсчёт количества рядов в выборке $result функцией
sql_num_rows(), как показано ниже:
if (sql_num_rows ($result) <= 0) {
print "Ошибка: не найдено ни одного ряда";
exit;
}
while ($row = sql_fetch_row ($result)){
print "$row[0]: $row[1]\n
\n";
}
?>
Избавляемся от do..while
Прежде всего, исчезает необходимость в использовании давно уже поднадоевшего
do..while, ибо для проверки на "ноль рядов" функция sql_num_row() не выдёргивает
первый рядв $row, и указатель по-прежнему установлен на начало.
В PHP Source как-то был представлен подобный фрагмент кода. Если выборка не была
нулевой, то функция sql_fetch_row() внутри условного блока доставляла первый
ряд. Для получения остальных приходилось прибегать к do..while, потому что
получение ряда из выборки ("to fetch" - принести, доставить// Прим. перев.)
смещает указатель в ней. Таким образом, сначала вам придётся обработать уже
полученный ряд ("do"), только потом получить второй ряд и так далее.
Так чем же do..while так провинился?
• в данном примере внутри цикла do..while помещён только один оператор: простой
вывод. Теперь представим, что там может оказаться не один, а десять операторов.
Тогда редактору кода придётся искать условие while после оператора do и целого
блока действий внутри цикла. Занятие не из приятных.
• условие while обычно располагается в начале блока, а не в конце его. Поэтому
редактору кода нужно будет уделять этому особое внимание при чтении, чтобы не
спутать цикл do..while с предварительным условием while обычного цикла.
Делаем всё просто и понятно
В случае получения нулевой выборки, функция sql_num_row() в отличие от
sql_fetch_row() делает именно то, что вам нужно сделать:
• действие sql_fetch_row(): "При попытке получить первый ряд не найдено ни
одного ряда. Это может означать, что в данной выборке их нет".
• Действие sql_num_row(): "Количество рядов в выборке равно нулю".
Но как это отражается на написании кода?
Рассмотрим следующий пример, где операторы внутри условия записаны псевдокодом:
• if(!($row = sql_fetch_row($result))){Print Error}:
• Получаем первый ряд из выборки.
• Если выборка пустая, то переменной $row приписываем 0; ноль логически
выражается False; отсюда !(0)=True; выводим сообщение об ошибке.
• Иначе, если выборка не пустая, получаем первый ряд, приписываем его переменной
$row; $row не равно нулю, то есть True; !(True)=False; выходим на цикл
do..while.
• If(sql_num_rows($result)<=0){Print Error}:
• Подсчёт рядов в выборке.
• Если их меньше или равно нулю, выводим сообщение об ошибке.
• Иначе - идём дальше.
Итак, какое из двух выражений проще и быстрее понять? Безусловно, подсчёт рядов
- более прямой и короткий путь.
Каково всё же практическое преимущество второго способа? Невелика разница, что
мы поместим внутри этого условия - многого тут не выиграть.
Однако на протяжении 10 000 строк вашего кода продуманные, а потому просто и
ясно изложенные идеи сэкономят кучу времени редактору кода (вот и первое
преимущество). Есть и другие преимущества: разработка скриптов заметно
ускоряется и становится более размеренной.
Если ваша СУБД не поддерживает sql_num_row()
Действительно, некоторые СУБД могут не поддерживать эту функцию. Отнесёмся с
сочувствием ко всем владельцам таких систем. Им придётся проверять выборки "на
ноль рядов" путем запроса первого ряда. Однако и здесь, рекомендуем использовать
булевские переменные:
$found = false;
while ($row = sql_fetch_array($result)){
$found = true;
}
if (!$found){
print "Ошибка";
}
?>
Получение рядов данных: правила эффективной работы
Вторая проблема нашего кода - это использование функции sql_fetch_row() для
получения рядов. Как результат своей работы эта функция возвращает лишь
пронумерованный массив. Однако существует ещё и функция sql_fetch_array(),
которая возвращает два массива: пронумерованный и ассоциативный:
$row = sql_fetch_array ($result);
print $row[1]; // Второй столбец
print $row[name]; // Столбец name - имя
?>
Примечание: Существуют разные точки зрения на целесообразность использования
одинарных кавычек при вставке строковых аргументов. В приведённом примере
(столбец name) и далее по статье они не используются.
Какая из функций более удобна для разработчика? Ассоциативные массивы позволяют
редактору кода ясно и однозначно понять, какая именно выборка из БД будет
осуществляться в каждом конкретном случае. Например:
if (sql_num_rows ($result) <= 0) {
print "Ошибка: не найдено ни одного ряда";
exit;
}
while ($row = sql_fetch_array ($result)) {
print "$row[name]: $row[phone_number]\n
\n";
}
?>
Применение sql_fetch_row($result)
Итак, функция sql_fetch_row() имеет целую тонну недостатков. Однако, существует
ситуация, где её можно поставить без всякого ущерба "прозрачности" кода: когда
sql-запрос формируется пользователем.
До настоящего момента мы рассматривали примеры с заранее известными запросами и
определёнными разработчиком. Но иногда возникает необходимость в запросе,
сформированном самим пользователем. В таких случаях разработчику неизвестно
количество столбцов в выборке.
Опубликовал Kest
October 26 2008 12:46:29 ·
1 Комментариев ·
65535 Прочтений ·
• Не нашли ответ на свой вопрос? Тогда задайте вопрос в комментариях или на форуме! •