This is an old revision of the document!
PHP RFC: println(string $data = ''): int
- Version: 0.1
- Date: 2021-03-13
- Author: Tyson Andre
- Status: Draft
- First Published at: http://wiki.php.net/rfc/println
Introduction
Printing a string followed by a newline to stdout is a commonly performed operation in many applications. Many programming languages provide a helper function to do this specifically, for readability and convenience. The choice of end of line may differ, but many recent programming languages will unconditionally use the unix newline, to avoid unexpected differences in behavior between platforms.
- Java has System.out.println('hello world')
- Python print('hello world')
prints a newline by default, and a named argument can override that.
- C has puts(“hello world”)
- Golang has fmt.Println(“hello world”)
- Rust has println!(“hello world”)
- And so on.
However, PHP currently does not yet have a standalone helper method to do this.
Proposal
Add a global function println(string $data = “”): int
to PHP.
The behavior is equivalent to the following polyfill but expected to be more efficient due to avoiding concatenation.
Similarly to https://www.php.net/manual/en/function.printf.php#refsect1-function.printf-returnvalues - this returns the number of bytes that were successfully output to stdout to be consistent with printf
. In the typical case where there was no output error, this returns strlen($data) + 1
. (E.g. printf and println may fail if php's standard output was redirected to a file on a disk that filled up)
/** * Prints $data followed by a unix newline * @return int the number of bytes that were successfully printed to stdout. */ function println(string $data = ''): int { return printf("%s\n", $data); } println("test"); println(); // moderately useful to not switch to echo or pass the empty string to print a blank line println("third line"); /* Output: test third line */
Reasons to Add This
- This is useful for self-contained scripts and a useful helper function to have overall. E.g. phpt tests of php itself print multiple lines for the
--EXPECT--
section, andvar_dump
can be overused even for known strings known not to have special characters or spaces becausevar_dump(some_function())
is a bit simpler to write thanecho some_function() . “\n”;
, but not as simple asprintln(some_function())
- Even if codebases add userland helper equivalents that do exactly this, If you are new to a codebase, or contribute to multiple codebases, it is inconvenient to use
xyz_println
,ABCUtils::println()
,echo X, “\n”
, etc., and remember if those different functions actually use the line endings you think they do.
Additionally, the prefixing is much more verbose. - In tutorials or language references that teach a developer how to use php functionality, it is often preferable to use functions that append a newline when multiple snippets would be evaluated together to keep examples simple.
println(“Hello $name”);
would be useful to have for introducing PHP to a new developer beforeecho “Hello $name\n”;
(requires explaining control characters first) orvar_dump(“Hello $name”);
(that debug representation is rarely useful forstring(11) “Hello world”
)
E.g.var_dump
is frequently used instead ofvar_export
,echo
, orprint
in the manual even for printing strings with no control characters such as https://www.php.net/manual/en/function.json-encode.php#example-3972
The Unix Newline is always used
This deliberately always prints the unix newline (\n
)
instead of PHP_EOL, for the reasons mentioned in this section.
I would find it very unexpected if println
were to behave
differently based on the web server was running it,
e.g. if you moved a website's backend from/to a Linux server
to/from a Windows server, responses generated by println would
suddenly be different. (Content-Length, hashes(e.g. sha256 sum) of output, etc.)
Additionally, https://www.php-fig.org/psr/psr-2/ recommends that all php
source files contain unix line endings.
If those files contain inline html/text snippets mixed with php+println(),
or if they contained strings using <<<EOT
heredoc,
it would be inconsistent to have \r\n
in the lines printed by
println() and \n
for heredoc and HTML when running on Windows.
The unix newline is same choice of line ending that var_dump
, debug_zval_dump
,
and var_export
use for dumping output.
Otherwise, println(“myArray=” . var_export($myArray, true));
would be a mix of multiple line ending choices.
PHP's interactive shell (php -a) also prints a single newline character if the output does not end in a newline.
// ext/readline/readline_cli.c if (!pager_pipe && php_last_char != '\0' && php_last_char != '\n') { php_write("\n", 1); }
Many new languages have elected to always use only the unix newlines, e.g. https://golang.org/pkg/fmt/#Println and https://doc.rust-lang.org/std/macro.println.html
Overall, editors do a much better job of detecting newline choices and displaying different newline choices than they did decades ago.
My opinion is that this anything generating files targeting a specific OS's line endings should continue to use PHP_EOL or continue to base the newline choice on the OS of the user requesting the output.
Type checking
Type checking is done the same way as other ordinary user-defined or internal global functions.
println(1);
or println(true)
would have the argument coerced to a string when strict_types
is disabled (the default), but would be a thrown Error
with strict_types=1
(like printf
would for the format string).
- Depending on the application, this may be useful as a runtime assertion or in making it clear to the reader that the argument is expected to be a string.
println((string)$value);
should be used when strict_types=1
but you are uncertain of the type.
(or other alternatives such as echo $arg, “\n”
, echo “$arg\n”;
, or printf(“%s\n”, $arg)
)
Backward Incompatible Changes
Declaring a function named println()
in the global namespace would become a duplicate function error.
Proposed PHP Version(s)
PHP 8.1
Unaffected PHP Functionality
Other printing functions or statements are unaffected. println
is NOT a keyword (e.g. functions named println can continue to be declared outside of the global namespace).
Future Scope
This section details areas where the feature might be improved in future, but that are not currently proposed in this RFC.
Proposed Voting Choices
Yes/No, requiring 2/3 majority.
References
https://externals.io/message/104545 “print with newline”