Почему две конструкции?
Правда о печати и эхо заключается в том, что, хотя они представляются пользователям в виде двух отдельных конструкций, они действительно являются оттенками эха, если вы приступите к основам, т. Е. Посмотрите на внутренний исходный код. Этот исходный код включает в себя как синтаксический анализатор, так и обработчики кода операции. Рассмотрим простое действие, например отображение нуля. Используете ли вы echo или print, будет вызываться один и тот же обработчик "ZEND_ECHO_SPEC_CONST_HANDLER". Обработчик для печати делает одну вещь, прежде чем он вызывает обработчик для эха, он гарантирует, что возвращаемое значение для печати равно 1, следующим образом:
ZVAL_LONG(&EX_T(opline->result.var).tmp_var, 1);
(см. здесь для справки )
Возвращаемое значение - удобство, если вы хотите использовать print в условном выражении. Почему 1, а не 100? В PHP истинность 1 или 100 одинакова, то есть истинна, тогда как 0 в логическом контексте приравнивается к ложному значению. В PHP все ненулевые значения (положительные и отрицательные) являются истинными значениями, и это вытекает из наследства PHP в Perl.
Но, если это так, то можно задаться вопросом, почему echo принимает несколько аргументов, тогда как print может обрабатывать только один. Для этого ответа нам нужно обратиться к парсеру, а именно к файлу zend_language_parser.y . Вы заметите, что в echo встроена гибкость, позволяющая печатать одно или несколько выражений (см. Здесь ). в то время как печать ограничена печатью только одного выражения (см. там ).
Синтаксис
В языке программирования C и языках, на которые он влияет, таких как PHP, существует различие между утверждениями и выражениями. Синтаксически, echo expr, expr, ... exprявляется оператором, а print exprявляется выражением, поскольку оно оценивается как значение. Поэтому, как и другие утверждения, echo exprстоит сам по себе и не может быть включен в выражение:
5 + echo 6; // syntax error
Напротив, print exprможет самостоятельно сформировать утверждение:
print 5; // valid
Или быть частью выражения:
$x = (5 + print 5); // 5
var_dump( $x ); // 6
Кто-то может подумать, printчто это унарный оператор, такой как !или, тем не ~менее, не оператор. Что !, ~ and printобъединяет то , что все они построены на PHP и каждый принимает только один аргумент. Вы можете использовать printдля создания следующего странного, но действительного кода:
<?php
print print print print 7; // 7111
На первый взгляд результат может показаться странным, что последний оператор print сначала печатает свой операнд '7' . Но если вы копаете глубже и смотрите на действительные коды операций, это имеет смысл:
line # * op fetch ext return operands
---------------------------------------------------------------------------------
3 0 > PRINT ~0 7
1 PRINT ~1 ~0
2 PRINT ~2 ~1
3 PRINT ~3 ~2
4 FREE ~3
5 > RETURN 1
Самый первый код операции, который генерируется, соответствует «print 7». «~ 0» - это временная переменная, значение которой равно 1. Эта переменная становится операндом для следующего кода операции печати, который, в свою очередь, возвращает временную переменную, и процесс повторяется. Последняя временная переменная вообще не используется, поэтому она освобождается.
Почему printвозвращает значение, а echoнет?
Выражения оценивают до значений. Например, 2 + 3оценивает 5и abs(-10)оценивает 10. Поскольку print exprсамо по себе является выражением, то оно должно содержать значение, и оно имеет, непротиворечивое значение 1указывает на достоверный результат, и, возвращая ненулевое значение, выражение становится полезным для включения в другое выражение. Например, в этом фрагменте возвращаемое значение print полезно при определении последовательности функций:
<?php
function bar( $baz ) {
// other code
}
function foo() {
return print("In and out ...\n");
}
if ( foo() ) {
bar();
}
Вы можете найти печать определенного значения, когда дело доходит до отладки на лету, как показано в следующем примере:
<?php
$haystack = 'abcde';
$needle = 'f';
strpos($haystack,$needle) !== FALSE OR print "$needle not in $haystack";
// output: f not in abcde
Как примечание, как правило, заявления не являются выражениями; они не возвращают значение. Исключением, конечно же, являются операторы выражений, которые используют print и даже простые выражения, используемые в качестве операторов, такие как1; синтаксис, который PHP наследует от C. Оператор выражения может выглядеть странно, но он очень полезен, позволяя передавать аргументы функции.
Это printфункция?
Нет, это языковая конструкция. Хотя все вызовы функций являются выражениями, print (expr)это выражение, несмотря на то, что визуальный элемент выглядит так, как если бы он использовал синтаксис вызова функций. По правде говоря, эти скобки представляют собой синтаксис скобок-expr, полезный для вычисления выражений. Это объясняет тот факт, что иногда они являются необязательными, если выражение является простым, например print "Hello, world!". С более сложным выражением, таким как print (5 ** 2 + 6/2); // 28скобки, можно помочь в оценке выражения. В отличие от имен функций, printсинтаксически является ключевым словом и семантически «языковой конструкцией» .
Термин «языковая конструкция» в PHP обычно относится к «псевдо» функциям, таким как issetили empty. Хотя эти «конструкции» выглядят в точности как функции, они на самом деле являются fexprs , то есть аргументы передаются им без оценки, что требует особой обработки от компилятора. printслучается, fexpr, который выбирает для оценки своего аргумента так же, как функции.
Разницу можно увидеть по распечатке get_defined_functions(): в printсписке нет функций. (Хотя printfи друзья: в отличие от них print, они истинные функции.)
Почему print (foo) работает тогда?
По той же причине, что echo(foo)работает. Эти круглые скобки сильно отличаются от круглых скобок вызова функций, потому что вместо этого они относятся к выражениям. Вот почему можно кодировать echo ( 5 + 8 )и ожидать результата 13 (см. Ссылку ). Эти круглые скобки участвуют в оценке выражения, а не в вызове функции. Примечание: в PHP есть и другие варианты использования скобок, такие как условные выражения if, списки присваиваний, объявления функций и т. Д.
Почему print(1,2,3)и echo(1,2,3)привести к ошибкам синтаксиса?
Синтаксис есть print expr, echo exprили echo expr, expr, ..., expr. Когда PHP встречается (1,2,3), он пытается проанализировать его как одно выражение и завершается неудачно, потому что в отличие от C, PHP на самом деле не имеет двоичного оператора запятой; запятая служит в качестве разделителя. (Тем не менее, вы можете найти двоичную запятую в циклах for PHP, синтаксис которой унаследован от C.)
Семантика
Это утверждение echo e1, e2, ..., eN;можно понимать как синтаксический сахар для echo e1; echo e2; ...; echo eN;.
Поскольку все выражения являются операторами и echo eвсегда имеют те же побочные эффекты print e, что и возвращаемое значение print eигнорируется при использовании в качестве оператора, мы можем понимать его echo eкак синтаксический сахар для print e.
Эти два наблюдения означают, что echo e1, e2, ..., eN;можно рассматривать как синтаксический сахар для print e1; print e2; ... print eN;. (Тем не менее, обратите внимание на несемантические различия во время выполнения ниже.)
Поэтому нам нужно только определить семантику для print. print eпри оценке:
- оценивает свой единственный аргумент
eи преобразует полученное значение в строку s. (Таким образом, print eэквивалентно print (string) e.)
- Потоки строку
sв выходной буфер (который в конечном счете будет передаваться на стандартный вывод).
- Оценивает до целого числа
1.
Различия на уровне байт-кода
print включает небольшие накладные расходы на заполнение возвращаемой переменной (псевдокод)
print 125;
PRINT 125,$temp ; print 125 and place 1 in $temp
UNSET $temp ; remove $temp
один echoкомпилируется в один код операции:
echo 125;
ECHO 125
многозначные echoкомпилируется в несколько операционных кодов
echo 123, 456;
ECHO 123
ECHO 456
Обратите внимание, что multi-value echoне объединяет свои аргументы, а выводит их один за другим.
Справка: zend_do_print, zend_do_echo.
Различия во время выполнения
ZEND_PRINT реализован следующим образом (псевдокод)
PRINT var, result:
result = 1
ECHO var
Таким образом, он в основном помещает 1переменную результата и делегирует реальную работу ZEND_ECHOобработчику. ZEND_ECHOделает следующее
ECHO var:
if var is object
temp = var->toString()
zend_print_variable(temp)
else
zend_print_variable(var)
где zend_print_variable()выполняет фактическую «печать» (фактически она просто перенаправляет на выделенную функцию SAPI).
Скорость: echo xпротивprint x
В отличие от echo , print выделяет временную переменную. Тем не менее, количество времени, затрачиваемого на эту деятельность, незначительно, поэтому разница между этими двумя языковыми конструкциями незначительна.
Скорость: echo a,b,cпротивecho a.b.c
Первый сводится к трем отдельным утверждениям. Второй оценивает все выражение a.b.c., печатает результат и сразу же удаляет его. Поскольку объединение включает в себя распределение памяти и копирование, первый вариант будет более эффективным.
Так какой из них использовать?
В веб-приложениях вывод в основном сконцентрирован в шаблонах. Поскольку шаблоны используют <?=псевдоним echo, кажется логичным придерживаться и echoдругих частей кода. echoимеет дополнительное преимущество, заключающееся в возможности печатать несколько выражений без их объединения, и не требует дополнительных затрат на заполнение временной возвращаемой переменной. Итак, пользуйтесь echo.