2016-08-19 16:33:59 +00:00
|
|
|
<!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01//EN"
|
|
|
|
"http://www.w3.org/TR/html4/strict.dtd">
|
|
|
|
<html>
|
|
|
|
<head>
|
|
|
|
<meta http-equiv="Content-Type" content="text/html;charset=US-ASCII">
|
|
|
|
<title>Text Formatting</title>
|
2016-08-19 23:43:10 +00:00
|
|
|
|
|
|
|
<style type="text/css">
|
|
|
|
|
|
|
|
body { color: #000000; background-color: #FFFFFF; }
|
|
|
|
del { text-decoration: line-through; color: #8B0040; }
|
|
|
|
ins { text-decoration: underline; color: #005100; }
|
|
|
|
|
|
|
|
p.example { margin-left: 2em; }
|
|
|
|
pre.example { margin-left: 2em; }
|
|
|
|
div.example { margin-left: 2em; }
|
|
|
|
|
|
|
|
code.extract { background-color: #F5F6A2; }
|
|
|
|
pre.extract { margin-left: 2em; background-color: #F5F6A2;
|
|
|
|
border: 1px solid #E1E28E; }
|
|
|
|
|
|
|
|
p.function { }
|
|
|
|
.attribute { margin-left: 2em; }
|
|
|
|
.attribute dt { float: left; font-style: italic;
|
|
|
|
padding-right: 1ex; }
|
|
|
|
.attribute dd { margin-left: 0em; }
|
|
|
|
|
|
|
|
blockquote.std { color: #000000; background-color: #F1F1F1;
|
|
|
|
border: 1px solid #D1D1D1;
|
|
|
|
padding-left: 0.5em; padding-right: 0.5em; }
|
|
|
|
blockquote.stddel { text-decoration: line-through;
|
|
|
|
color: #000000; background-color: #FFEBFF;
|
|
|
|
border: 1px solid #ECD7EC;
|
|
|
|
padding-left: 0.5empadding-right: 0.5em; ; }
|
|
|
|
|
|
|
|
blockquote.stdins { text-decoration: underline;
|
|
|
|
color: #000000; background-color: #C8FFC8;
|
|
|
|
border: 1px solid #B3EBB3; padding: 0.5em; }
|
|
|
|
|
|
|
|
table { border: 1px solid black; border-spacing: 0px;
|
|
|
|
margin-left: auto; margin-right: auto; }
|
|
|
|
th { text-align: left; vertical-align: top;
|
|
|
|
padding-left: 0.8em; border: none; }
|
|
|
|
td { text-align: left; vertical-align: top;
|
|
|
|
padding-left: 0.8em; border: none; }
|
|
|
|
|
|
|
|
</style>
|
|
|
|
|
2016-08-24 13:55:18 +00:00
|
|
|
<script src="https://ajax.googleapis.com/ajax/libs/jquery/1.7.1/jquery.min.js"
|
|
|
|
type="text/javascript"> </script>
|
|
|
|
|
|
|
|
<script type="text/javascript">$(function() {
|
|
|
|
var next_id = 0
|
|
|
|
function find_id(node) {
|
|
|
|
// Look down the first children of 'node' until we find one
|
|
|
|
// with an id. If we don't find one, give 'node' an id and
|
|
|
|
// return that.
|
|
|
|
var cur = node[0];
|
|
|
|
while (cur) {
|
|
|
|
if (cur.id) return curid;
|
|
|
|
if (cur.tagName == 'A' && cur.name)
|
|
|
|
return cur.name;
|
|
|
|
cur = cur.firstChild;
|
|
|
|
};
|
|
|
|
// No id.
|
|
|
|
node.attr('id', 'gensection-' + next_id++);
|
|
|
|
return node.attr('id');
|
|
|
|
};
|
|
|
|
|
|
|
|
// Put a table of contents in the #toc nav.
|
|
|
|
|
|
|
|
// This is a list of <ol> elements, where toc[N] is the list for
|
|
|
|
// the current sequence of <h(N+2)> tags. When a header of an
|
|
|
|
// existing level is encountered, all higher levels are popped,
|
|
|
|
// and an <li> is appended to the level
|
|
|
|
var toc = [$("<ol/>")];
|
|
|
|
$(':header').not('h1').each(function() {
|
|
|
|
var header = $(this);
|
|
|
|
// For each <hN> tag, add a link to the toc at the appropriate
|
|
|
|
// level. When toc is one element too short, start a new list
|
|
|
|
var levels = {H2: 0, H3: 1, H4: 2, H5: 3, H6: 4};
|
|
|
|
var level = levels[this.tagName];
|
|
|
|
if (typeof level == 'undefined') {
|
|
|
|
throw 'Unexpected tag: ' + this.tagName;
|
|
|
|
}
|
|
|
|
// Truncate to the new level.
|
|
|
|
toc.splice(level + 1, toc.length);
|
|
|
|
if (toc.length < level) {
|
|
|
|
// Omit TOC entries for skipped header levels.
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if (toc.length == level) {
|
|
|
|
// Add a <ol> to the previous level's last <li> and push
|
|
|
|
// it into the array.
|
|
|
|
var ol = $('<ol/>')
|
|
|
|
toc[toc.length - 1].children().last().append(ol);
|
|
|
|
toc.push(ol);
|
|
|
|
}
|
|
|
|
var header_text = header.text();
|
|
|
|
toc[toc.length - 1].append(
|
|
|
|
$('<li/>').append($('<a href="#' + find_id(header) + '"/>')
|
|
|
|
.text(header_text)));
|
|
|
|
});
|
|
|
|
$('#toc').append(toc[0]);
|
|
|
|
})
|
|
|
|
</script>
|
|
|
|
|
2016-08-19 16:33:59 +00:00
|
|
|
</head>
|
|
|
|
<body>
|
|
|
|
<h1>Text Formatting</h1>
|
|
|
|
|
|
|
|
<p>
|
|
|
|
2016-08-19
|
|
|
|
</p>
|
|
|
|
|
|
|
|
<address>
|
|
|
|
Victor Zverovich, victor.zverovich@gmail.com
|
|
|
|
</address>
|
|
|
|
|
2016-08-24 13:55:18 +00:00
|
|
|
<div id="toc"></div>
|
2016-08-19 16:33:59 +00:00
|
|
|
|
|
|
|
<h2><a name="Introduction">Introduction</a></h2>
|
|
|
|
|
|
|
|
<p>
|
|
|
|
This paper proposes a new text formatting functionality that can be used as a
|
|
|
|
safe and extensible alternative to the <code>printf</code> family of functions.
|
|
|
|
It is intended to complement the existing C++ I/O streams library and reuse
|
|
|
|
some of its infrastructure such as overloaded insertion operators for
|
|
|
|
user-defined types.
|
|
|
|
</p>
|
|
|
|
|
|
|
|
<p>
|
|
|
|
Example:
|
|
|
|
|
2016-08-19 23:43:10 +00:00
|
|
|
<pre class="example">
|
|
|
|
<code>std::string message = std::format("The answer is {}.", 42);</code>
|
2016-08-19 16:33:59 +00:00
|
|
|
</pre>
|
|
|
|
|
|
|
|
<h2><a name="Design">Design</a></h2>
|
|
|
|
|
|
|
|
<h3><a name="Syntax">Format String Syntax</a></h3>
|
|
|
|
|
|
|
|
<p>
|
|
|
|
Variations of the printf format string syntax are arguably the most popular
|
|
|
|
among the programming languages and C++ itself inherits <code>printf</code>
|
|
|
|
from C <a href="#1">[1]</a>. The advantage of the printf syntax is that many
|
|
|
|
programmers are familiar with it. However, in its current form it has a number
|
|
|
|
of issues:
|
|
|
|
</p>
|
|
|
|
|
|
|
|
<ul>
|
|
|
|
<li>Many format specifiers like <code>hh</code>, <code>h</code>, <code>l</code>,
|
|
|
|
<code>j</code>, etc. are used only to convey type information.
|
|
|
|
They are redundant in type-safe formatting and would unnecessarily
|
|
|
|
complicate specification and parsing.</li>
|
|
|
|
<li>There is no standard way to extend the syntax for user-defined types.</li>
|
|
|
|
<li>There are subtle differences between different implementations. For example,
|
|
|
|
POSIX positional arguments <a href="#2">[2]</a> are not supported on
|
2016-08-19 23:43:10 +00:00
|
|
|
some systems <a href="#6">[6]</a>.</li>
|
2016-08-19 16:33:59 +00:00
|
|
|
<li>Using <code>'%'</code> in a custom format specifier, e.g. for
|
|
|
|
<code>put_time</code>-like time formatting, poses difficulties.</li>
|
|
|
|
</ul>
|
|
|
|
|
|
|
|
<p>
|
|
|
|
Although it is possible to address these issues, this will break compatibility
|
|
|
|
and can potentially be more confusing to users than introducing a different
|
|
|
|
syntax.
|
|
|
|
</p>
|
|
|
|
|
2016-08-24 14:41:07 +00:00
|
|
|
<p>
|
2016-08-19 16:33:59 +00:00
|
|
|
Therefore we propose a new syntax based on the ones used in Python
|
|
|
|
<a href="#3">[3]</a>, the .NET family of languages <a href="#4">[4]</a>,
|
2016-08-20 04:41:19 +00:00
|
|
|
and Rust <a href="#5">[5]</a>. This syntax employs <code>'{'</code> and
|
2016-08-19 16:33:59 +00:00
|
|
|
<code>'}'</code> as replacement field delimiters instead of <code>'%'</code>
|
2016-08-23 15:10:35 +00:00
|
|
|
and it is described in details in the <a href="#SyntaxRef">syntax reference</a>.
|
|
|
|
Here are some of the advantages:
|
2016-08-19 16:33:59 +00:00
|
|
|
</p>
|
|
|
|
|
|
|
|
<ul>
|
|
|
|
<li>Consistent and easy to parse mini-language focused on formatting rather
|
|
|
|
than conveying type information</li>
|
|
|
|
<li>Extensibility and support for custom format strings for user-defined
|
|
|
|
types</li>
|
|
|
|
<li>Positional arguments</li>
|
|
|
|
<li>Support for both locale-specific and locale-independent formatting (see
|
|
|
|
<a href="#Locale">Locale Support</a>)</li>
|
2016-08-21 14:50:31 +00:00
|
|
|
<li>Formatting improvements such as better alignment control, fill character,
|
|
|
|
and binary format
|
2016-08-19 16:33:59 +00:00
|
|
|
</ul>
|
|
|
|
|
|
|
|
<p>
|
|
|
|
The syntax is expressive enough to enable translation, possibly automated,
|
2016-08-19 23:43:10 +00:00
|
|
|
of most printf format strings. The correspondence between <code>printf</code>
|
|
|
|
and the new syntax is given in the following table.
|
|
|
|
</p>
|
|
|
|
|
|
|
|
<table>
|
|
|
|
<thead>
|
2016-08-21 14:50:31 +00:00
|
|
|
<tr><th>printf</th><th>new</th></tr>
|
2016-08-19 23:43:10 +00:00
|
|
|
</thead>
|
|
|
|
<tbody>
|
2016-08-21 14:50:31 +00:00
|
|
|
<tr><td>-</td><td><</td></tr>
|
|
|
|
<tr><td>+</td><td>+</td></tr>
|
|
|
|
<tr><td><em>space</em></td><td><em>space</em></td></tr>
|
|
|
|
<tr><td>#</td><td>#</td></tr>
|
|
|
|
<tr><td>0</td><td>0</td></tr>
|
|
|
|
<tr><td>hh</td><td>unused</td></tr>
|
|
|
|
<tr><td>h</td><td>unused</td></tr>
|
|
|
|
<tr><td>l</td><td>unused</td></tr>
|
|
|
|
<tr><td>ll</td><td>unused</td></tr>
|
|
|
|
<tr><td>j</td><td>unused</td></tr>
|
|
|
|
<tr><td>z</td><td>unused</td></tr>
|
|
|
|
<tr><td>t</td><td>unused</td></tr>
|
|
|
|
<tr><td>L</td><td>unused</td></tr>
|
|
|
|
<tr><td>c</td><td>c (optional)</td></tr>
|
|
|
|
<tr><td>s</td><td>s (optional)</td></tr>
|
|
|
|
<tr><td>d</td><td>d (optional)</td></tr>
|
|
|
|
<tr><td>i</td><td>d (optional)</td></tr>
|
|
|
|
<tr><td>o</td><td>o</td></tr>
|
|
|
|
<tr><td>x</td><td>x</td></tr>
|
|
|
|
<tr><td>X</td><td>X</td></tr>
|
|
|
|
<tr><td>u</td><td>d (optional)</td></tr>
|
|
|
|
<tr><td>f</td><td>f</td></tr>
|
|
|
|
<tr><td>F</td><td>F</td></tr>
|
|
|
|
<tr><td>e</td><td>e</td></tr>
|
|
|
|
<tr><td>E</td><td>E</td></tr>
|
|
|
|
<tr><td>a</td><td>a</td></tr>
|
|
|
|
<tr><td>A</td><td>A</td></tr>
|
|
|
|
<tr><td>g</td><td>g (optional)</td></tr>
|
|
|
|
<tr><td>G</td><td>G</td></tr>
|
|
|
|
<tr><td>n</td><td>unused</td></tr>
|
|
|
|
<tr><td>p</td><td>p (optional)</td></tr>
|
2016-08-19 23:43:10 +00:00
|
|
|
</tbody>
|
|
|
|
</table>
|
|
|
|
|
|
|
|
<p>
|
|
|
|
Width and precision are represented similarly in <code>printf</code> and the
|
|
|
|
proposed syntax with the only difference that runtime value is specified by
|
|
|
|
<code>*</code> in the former and <code>{}</code> in the latter, possibly with
|
|
|
|
the index of the argument inside the braces.
|
|
|
|
</p>
|
|
|
|
|
|
|
|
<p>
|
|
|
|
As can be seen from the table above, most of the specifiers remain the same
|
2016-08-20 14:39:10 +00:00
|
|
|
which simplifies migration from <code>printf</code>. Notable difference is
|
|
|
|
in the alignment specification. The proposed syntax allows left, center,
|
|
|
|
and right alignment represented by <code>'<'</code>, <code>'^'</code>,
|
|
|
|
and <code>'>'</code> respectively which is more expressive than the
|
|
|
|
corresponding <code>printf</code> syntax. The latter only supports left and
|
|
|
|
right (the default) alignment.
|
2016-08-20 04:41:19 +00:00
|
|
|
</p>
|
|
|
|
|
|
|
|
<p>
|
|
|
|
The following example uses center alignment and <code>'*'</code> as a fill
|
|
|
|
character:
|
|
|
|
</p>
|
|
|
|
|
|
|
|
<pre class="example">
|
|
|
|
<code>std::format("{:*^30}", "centered");</code>
|
|
|
|
</pre>
|
|
|
|
|
|
|
|
<p>
|
|
|
|
resulting in <code>"***********centered***********"</code>.
|
|
|
|
The same formatting cannot be easily achieved with <code>printf</code>.
|
2016-08-19 23:43:10 +00:00
|
|
|
</p>
|
|
|
|
|
|
|
|
<h3><a name="Extensibility">Extensibility</a></h3>
|
|
|
|
|
|
|
|
<p>
|
2016-08-20 14:39:10 +00:00
|
|
|
Both the format string syntax and the API are designed with extensibility in
|
|
|
|
mind. The mini-language can be extended for user-defined types and users can
|
|
|
|
provide functions that do parsing and formatting for such types.
|
2016-08-19 16:33:59 +00:00
|
|
|
</p>
|
|
|
|
|
2016-08-19 23:43:10 +00:00
|
|
|
<p>The general syntax of a replacement field in a format string is
|
|
|
|
|
2016-08-21 14:50:31 +00:00
|
|
|
<pre>
|
|
|
|
<code>replacement-field ::= '{' [arg-id] [':' format-spec] '}'</code>
|
|
|
|
</pre>
|
2016-08-19 23:43:10 +00:00
|
|
|
|
|
|
|
<p>
|
2016-08-21 14:50:31 +00:00
|
|
|
where <code>format-spec</code> is predefined for built-in types, but can be
|
2016-08-20 04:41:19 +00:00
|
|
|
customized for user-defined types. For example, the syntax can be extended
|
2016-08-20 14:39:10 +00:00
|
|
|
for <code>put_time</code>-like date and time formatting
|
2016-08-20 04:41:19 +00:00
|
|
|
</p>
|
|
|
|
|
|
|
|
<pre class="example">
|
|
|
|
<code>std::time_t t = std::time(nullptr);
|
2016-08-24 14:41:07 +00:00
|
|
|
std::string date = std::format("The date is {0:%Y-%m-%d}.", *std::localtime(&t));</code>
|
2016-08-20 04:41:19 +00:00
|
|
|
</pre>
|
2016-08-19 23:43:10 +00:00
|
|
|
|
2016-08-20 14:39:10 +00:00
|
|
|
<p>by providing an overload of <code>std::format_arg</code> for
|
|
|
|
<code>std::tm</code>:</p>
|
|
|
|
|
2016-08-24 14:41:07 +00:00
|
|
|
<p>TODO: example</p>
|
2016-08-20 14:39:10 +00:00
|
|
|
|
|
|
|
<h3><a name="Safety">Safety</a></h3>
|
|
|
|
|
2016-08-24 14:41:07 +00:00
|
|
|
<p>
|
2016-08-20 14:39:10 +00:00
|
|
|
Formatting functions rely on variadic templates instead of the mechanism
|
|
|
|
provided by <code><cstdarg></code>. The type information is captured
|
|
|
|
automatically and passed to formatters guaranteeing type safety and making
|
|
|
|
many of the <code>printf</code> specifiers redundant (see <a href="#Syntax">
|
|
|
|
Format String Syntax</a>). Buffer management is also automatic to prevent
|
|
|
|
buffer overflow errors common to <code>printf</code>.
|
2016-08-24 14:41:07 +00:00
|
|
|
</p>
|
2016-08-19 23:43:10 +00:00
|
|
|
|
2016-08-19 16:33:59 +00:00
|
|
|
<h3><a name="Locale">Locale Support</a></h3>
|
|
|
|
|
2016-08-20 14:39:10 +00:00
|
|
|
<p>
|
|
|
|
As pointed out in
|
|
|
|
<a href="http://www.open-std.org/jtc1/sc22/wg21/docs/papers/2016/p0067r1.html">
|
|
|
|
P0067R1: Elementary string conversions</a> there is a number of use
|
|
|
|
cases that do not require internationalization support, but do require high
|
|
|
|
throughput when produced by a server. These include various text-based
|
|
|
|
interchange formats such as JSON or XML. The need for locale-independent
|
|
|
|
functions for conversions between integers and strings and between
|
|
|
|
floating-point numbers and strings has also been highlighted in
|
|
|
|
<a href="http://open-std.org/JTC1/SC22/WG21/docs/papers/2015/n4412.html">
|
|
|
|
N4412: Shortcomings of iostreams</a>. Therefore a user should be able to
|
|
|
|
easily control whether to use locales or not during formatting.
|
|
|
|
</p>
|
|
|
|
|
|
|
|
<p>
|
|
|
|
We follow Python's approach <a href="#3">[3]</a> and designate a separate format
|
|
|
|
specifier <code>'n'</code> for locale-aware numeric formatting. It applies to
|
|
|
|
all integral and floating-point types. All other specifiers produce output
|
2016-08-21 14:50:31 +00:00
|
|
|
unaffected by locale settings. This can also have positive peformance effect
|
|
|
|
because locale-independent formatting can be implemented more efficiently.
|
2016-08-20 14:39:10 +00:00
|
|
|
</p>
|
2016-08-19 16:33:59 +00:00
|
|
|
|
2016-08-20 04:41:19 +00:00
|
|
|
<h3><a name="PosArguments">Positional Arguments</a></h3>
|
|
|
|
|
2016-08-21 14:50:31 +00:00
|
|
|
<p>
|
|
|
|
An important feature for localization is the ability to rearrange formatting
|
|
|
|
arguments because the word order may vary in different languages
|
|
|
|
<a href="#3">[3]</a>. For example:
|
|
|
|
</p>
|
|
|
|
|
|
|
|
<pre class="example">
|
2016-08-23 15:10:35 +00:00
|
|
|
<code>printf("String `%s' has %d characters\n", string, length(string)));</code>
|
2016-08-21 14:50:31 +00:00
|
|
|
</pre>
|
|
|
|
|
|
|
|
<p>A possible German translation of the format string might be:</p>
|
|
|
|
|
|
|
|
<pre class="example">
|
|
|
|
<code>"%2$d Zeichen lang ist die Zeichenkette `%1$s'\n"</code>
|
|
|
|
</pre>
|
|
|
|
|
|
|
|
<p>
|
|
|
|
using POSIX positional arguments <a href="#2">[2]</a>. Unfortunately these
|
|
|
|
positional specifiers are not portable <a href="#6">[6]</a>. The C++ I/O
|
2016-08-23 15:10:35 +00:00
|
|
|
streams don't support such rearranging of arguments by design because they
|
|
|
|
are interleaved with the portions of the literal string:
|
2016-08-21 14:50:31 +00:00
|
|
|
</p>
|
|
|
|
|
|
|
|
<pre class="example">
|
2016-08-23 15:10:35 +00:00
|
|
|
<code>std::cout << "String `" << string << "' has " << length(string) << " characters\n";</code>
|
2016-08-21 14:50:31 +00:00
|
|
|
</pre>
|
|
|
|
|
|
|
|
<p>
|
|
|
|
The current proposal allows both positional and automatically numbered
|
|
|
|
arguments, for example:
|
|
|
|
</p>
|
|
|
|
|
|
|
|
<pre class="example">
|
2016-08-23 15:10:35 +00:00
|
|
|
<code>std::format("String `{}' has {} characters\n", string, length(string)));</code>
|
2016-08-21 14:50:31 +00:00
|
|
|
</pre>
|
|
|
|
|
|
|
|
<p>with the German translation of the format string:</p>
|
|
|
|
|
|
|
|
<pre class="example">
|
|
|
|
<code>"{1} Zeichen lang ist die Zeichenkette `{0}'\n"</code>
|
|
|
|
</pre>
|
2016-08-20 04:41:19 +00:00
|
|
|
|
2016-08-20 14:39:10 +00:00
|
|
|
<h3><a name="Locale">Performance</a></h3>
|
|
|
|
|
|
|
|
<p>TODO</p>
|
|
|
|
|
2016-08-21 14:50:31 +00:00
|
|
|
<h3><a name="Footprint">Binary Footprint</a></h3>
|
2016-08-19 16:33:59 +00:00
|
|
|
|
|
|
|
<p>TODO</p>
|
|
|
|
|
2016-08-21 14:50:31 +00:00
|
|
|
<h2><a name="Wording">Proposed Wording</a></h2>
|
|
|
|
|
|
|
|
<h3>Header <code><format></code> synopsis</h3>
|
|
|
|
|
|
|
|
<pre>
|
|
|
|
<code>namespace std {
|
2016-08-23 15:10:35 +00:00
|
|
|
class format_error;
|
|
|
|
|
2016-08-21 14:50:31 +00:00
|
|
|
class format_args;
|
|
|
|
|
|
|
|
template <class Char>
|
2016-08-23 15:10:35 +00:00
|
|
|
basic_string<Char> format(const Char* fmt, format_args args);
|
2016-08-21 14:50:31 +00:00
|
|
|
|
|
|
|
template <class Char, class ...Args>
|
2016-08-23 15:10:35 +00:00
|
|
|
basic_string<Char> format(const Char* fmt, const Args&... args);
|
2016-08-21 14:50:31 +00:00
|
|
|
}</code>
|
|
|
|
</pre>
|
|
|
|
|
2016-08-23 15:10:35 +00:00
|
|
|
<h3><a name="SyntaxRef">Format string syntax</a></h3>
|
|
|
|
|
|
|
|
<p>
|
|
|
|
Format strings contain <em>replacement fields</em> surrounded by curly braces
|
|
|
|
<code>{}</code>. Anything that is not contained in braces is considered literal
|
|
|
|
text, which is copied unchanged to the output. A brace character can be
|
|
|
|
included in the literal text by doubling: <code>{{</code> and <code>}}</code>.
|
|
|
|
</p>
|
2016-08-21 14:50:31 +00:00
|
|
|
|
2016-08-24 13:55:18 +00:00
|
|
|
<p>
|
|
|
|
The grammar for a replacement field is as follows:
|
|
|
|
</p>
|
|
|
|
|
|
|
|
<!-- The notation is the same as in n4296 22.4.3.1. -->
|
2016-08-21 14:50:31 +00:00
|
|
|
<pre>
|
|
|
|
<code>replacement-field ::= '{' [arg-id] [':' format-spec] '}'
|
|
|
|
arg-id ::= integer
|
|
|
|
integer ::= digit+
|
2016-08-23 15:10:35 +00:00
|
|
|
digit ::= '0'...'9'</code>
|
2016-08-21 14:50:31 +00:00
|
|
|
</pre>
|
|
|
|
|
2016-08-24 13:55:18 +00:00
|
|
|
<p>
|
|
|
|
In less formal terms, the replacement field can start with an
|
|
|
|
<code>arg-id</code> that specifies the argument whose value is to be formatted
|
|
|
|
and inserted into the output instead of the replacement field. The
|
|
|
|
<code>arg-id</code> is optionally followed by a <code>format-spec</code>,
|
|
|
|
which is preceded by a colon <code>':'</code>. These specify a non-default
|
|
|
|
format for the replacement value.
|
|
|
|
</p>
|
|
|
|
|
|
|
|
<p>
|
|
|
|
See also the <a href="FormatSpec">Format specification mini-language</a>
|
|
|
|
section.
|
|
|
|
</p>
|
|
|
|
|
|
|
|
<p>
|
|
|
|
If the numerical <code>arg-id</code>s in a format string are 0, 1, 2, ... in
|
|
|
|
sequence, they can all be omitted (not just some) and the numbers 0, 1, 2, ...
|
|
|
|
will be automatically inserted in that order.
|
|
|
|
</p>
|
|
|
|
|
|
|
|
<p>
|
|
|
|
Some simple format string examples:
|
|
|
|
</p>
|
|
|
|
|
|
|
|
<pre>
|
|
|
|
<code>"First, thou shalt count to {0}" // References the first argument
|
|
|
|
"Bring me a {}" // Implicitly references the first argument
|
|
|
|
"From {} to {}" // Same as "From {0} to {1}"</code>
|
|
|
|
</pre>
|
|
|
|
|
|
|
|
<p>
|
|
|
|
The <code>format-spec</code> field contains a specification of how the value
|
|
|
|
should be presented, including such details as field width, alignment, padding,
|
|
|
|
decimal precision and so on. Each value type can define its own <em>formatting
|
|
|
|
mini-language</em> or interpretation of the <code>format-spec</code>.
|
|
|
|
</p>
|
|
|
|
|
|
|
|
<p>
|
|
|
|
Most built-in types support a common formatting mini-language, which is
|
|
|
|
described in the next section.
|
|
|
|
</p>
|
|
|
|
|
|
|
|
<p>
|
|
|
|
A <code>format-spec</code> field can also include nested replacement fields
|
2016-08-25 13:50:09 +00:00
|
|
|
in certain position within it. These nested replacement fields can contain only
|
|
|
|
an argument index; format specifications are not allowed. This allows the
|
|
|
|
formatting of a value to be dynamically specified.
|
2016-08-24 13:55:18 +00:00
|
|
|
</p>
|
|
|
|
|
|
|
|
<h4><a name="FormatSpec">Format specification mini-language</a></h4>
|
|
|
|
|
|
|
|
<p>
|
|
|
|
<em>Format specifications</em> are used within replacement fields contained
|
|
|
|
within a format string to define how individual values are presented (see
|
|
|
|
<a href="SyntaxRef">Format string syntax</a>). Each formattable type may define
|
|
|
|
how the format specification is to be interpreted.
|
|
|
|
</p>
|
|
|
|
|
|
|
|
<p>
|
|
|
|
Most built-in types implement the following options for format specifications,
|
|
|
|
although some of the formatting options are only supported by the numeric types.
|
|
|
|
</p>
|
|
|
|
|
|
|
|
<p>
|
|
|
|
The general form of a <em>standard format specifier</em> is:
|
|
|
|
</p>
|
|
|
|
|
2016-08-21 14:50:31 +00:00
|
|
|
<pre>
|
|
|
|
<code>format-spec ::= [[fill] align] [sign] ['#'] ['0'] [width] ['.' precision] [type]
|
|
|
|
fill ::= <a character other than '{' or '}'>
|
|
|
|
align ::= '<' | '>' | '=' | '^'
|
|
|
|
sign ::= '+' | '-' | ' '
|
|
|
|
width ::= integer | '{' arg-id '}'
|
|
|
|
precision ::= integer | '{' arg-id '}'
|
|
|
|
type ::= int-type | 'a' | 'A' | 'c' | 'e' | 'E' | 'f' | 'F' | 'g' | 'G' | 'p' | 's'
|
|
|
|
int-type ::= 'b' | 'B' | 'd' | 'o' | 'x' | 'X'</code>
|
|
|
|
</pre>
|
|
|
|
|
2016-08-24 14:41:07 +00:00
|
|
|
<p>
|
|
|
|
The <code>fill</code> character can be any character other than <code>'{'</code>
|
|
|
|
or <code>'}'</code>. The presence of a fill character is signaled by the
|
|
|
|
character following it, which must be one of the alignment options. If the
|
|
|
|
second character of <code>format-spec</code> is not a valid alignment option,
|
|
|
|
then it is assumed that both the fill character and the alignment option are
|
|
|
|
absent.
|
|
|
|
|
|
|
|
<p>
|
|
|
|
The meaning of the various alignment options is as follows:
|
|
|
|
</p>
|
|
|
|
|
|
|
|
<table>
|
|
|
|
<thead>
|
|
|
|
<tr><th>Option</th><th>Meaning</th></tr>
|
|
|
|
</thead>
|
|
|
|
<tbody>
|
|
|
|
<tr>
|
|
|
|
<td>'<'</td>
|
|
|
|
<td>Forces the field to be left-aligned within the available space (this is
|
|
|
|
the default for most objects).</td>
|
|
|
|
</tr>
|
|
|
|
<tr>
|
|
|
|
<td>'>'</td>
|
|
|
|
<td>Forces the field to be right-aligned within the available space (this is
|
|
|
|
the default for numbers).</td>
|
|
|
|
</tr>
|
|
|
|
<tr>
|
|
|
|
<td>'='</td>
|
|
|
|
<td>Forces the padding to be placed after the sign (if any) but before the
|
|
|
|
digits. This is used for printing fields in the form
|
|
|
|
<code>+000000120</code>. This alignment option is only valid for numeric
|
|
|
|
types.</td>
|
|
|
|
</tr>
|
|
|
|
<tr>
|
|
|
|
<td>'^'</td>
|
|
|
|
<td>Forces the field to be centered within the available space.</td>
|
|
|
|
</tr>
|
|
|
|
</tbody>
|
|
|
|
</table>
|
|
|
|
|
2016-08-25 13:50:09 +00:00
|
|
|
<p>
|
|
|
|
Note that unless a minimum field width is defined, the field width will always
|
|
|
|
be the same size as the data to fill it, so that the alignment option has no
|
|
|
|
meaning in this case.
|
|
|
|
</p>
|
|
|
|
|
|
|
|
<p>
|
|
|
|
The <code>sign</code> option is only valid for number types, and can be one of
|
|
|
|
the following:
|
|
|
|
</p>
|
|
|
|
|
|
|
|
<table>
|
|
|
|
<thead>
|
|
|
|
<tr><th>Option</th><th>Meaning</th></tr>
|
|
|
|
</thead>
|
|
|
|
<tbody>
|
|
|
|
<tr>
|
|
|
|
<td>'+'</td>
|
|
|
|
<td>Indicates that a sign should be used for both positive as well as negative
|
|
|
|
numbers.</td>
|
|
|
|
</tr>
|
|
|
|
<tr>
|
|
|
|
<td>'-'</td>
|
|
|
|
<td>Indicates that a sign should be used only for negative numbers (this is
|
|
|
|
the default behavior).</td>
|
|
|
|
</tr>
|
|
|
|
<tr>
|
|
|
|
<td>space</td>
|
|
|
|
<td>Indicates that a leading space should be used on positive numbers, and a
|
|
|
|
minus sign on negative numbers.</td>
|
|
|
|
</tr>
|
|
|
|
</tbody>
|
|
|
|
</table>
|
|
|
|
|
2016-08-24 14:41:07 +00:00
|
|
|
<p>TODO</p>
|
2016-08-24 13:55:18 +00:00
|
|
|
|
2016-08-23 15:10:35 +00:00
|
|
|
<h3>Class <code>format_error</code></h3>
|
|
|
|
|
|
|
|
<pre>
|
|
|
|
<code>class format_error : public std::runtime_error {
|
|
|
|
public:
|
|
|
|
explicit format_error(const string& what_arg);
|
|
|
|
explicit format_error(const char* what_arg);
|
|
|
|
};</code>
|
|
|
|
</pre>
|
|
|
|
|
|
|
|
<p>
|
|
|
|
The class <code>format_error</code> defines the type of objects thrown as
|
|
|
|
exceptions to report errors from the formatting library.
|
|
|
|
</p>
|
|
|
|
|
|
|
|
<dl>
|
|
|
|
<dt><code>format_error(const string& what_arg);</code></dt>
|
|
|
|
<dd>
|
|
|
|
<p><i>Effects</i>: Constructs an object of class <code>format_error</code>.</p>
|
|
|
|
<p><i>Postcondition</i>: <code>strcmp(what(), what_arg.c_str()) == 0</code>.</p>
|
|
|
|
</dd>
|
|
|
|
<dt><code>format_error(const char* what_arg);</code></dt>
|
|
|
|
<dd>
|
|
|
|
<p><i>Effects</i>: Constructs an object of class <code>format_error</code>.</p>
|
|
|
|
<p><i>Postcondition</i>: <code>strcmp(what(), what_arg) == 0</code>.</p>
|
|
|
|
</dd>
|
2016-08-24 14:41:07 +00:00
|
|
|
</dl>
|
2016-08-23 15:10:35 +00:00
|
|
|
|
2016-08-24 13:55:18 +00:00
|
|
|
<h3>Class <code>format_args</code></h3>
|
|
|
|
|
|
|
|
<p>TODO</p>
|
|
|
|
|
2016-08-23 15:10:35 +00:00
|
|
|
<h3>Function template <code>format</code></h3>
|
|
|
|
|
|
|
|
<dl>
|
|
|
|
<dt>
|
2016-08-24 14:41:07 +00:00
|
|
|
<code>template <class Char><br>
|
|
|
|
basic_string<Char> format(const Char* fmt, format_args args);<br>
|
|
|
|
<br>
|
|
|
|
template <class Char, class ...Args><br>
|
|
|
|
basic_string<Char> format(const Char* fmt, const Args&... args);</code>
|
2016-08-23 15:10:35 +00:00
|
|
|
</dt>
|
|
|
|
<dd>
|
|
|
|
|
|
|
|
<p><i>Requires</i>: <code>fmt</code> shall not be a null pointer.</p>
|
|
|
|
<p>
|
|
|
|
<i>Effects</i>: Each function returns a <code>basic_string</code> object
|
|
|
|
constructed from the format string argument <code>fmt</code> with each
|
|
|
|
replacement field substituted with the character representation of the
|
|
|
|
argument it refers to, formatted according to the specification given in the
|
|
|
|
field.
|
|
|
|
</p>
|
|
|
|
<p><i>Returns</i>: The formatted string.</p>
|
|
|
|
<p><i>Throws</i>: <code>format_error</code> if <code>fmt</code> is not a valid
|
|
|
|
format string.</p>
|
|
|
|
</dd>
|
2016-08-24 14:41:07 +00:00
|
|
|
</dl>
|
2016-08-23 15:10:35 +00:00
|
|
|
|
2016-08-19 16:33:59 +00:00
|
|
|
<h2><a name="Implementation">Implementation</a></h2>
|
|
|
|
|
|
|
|
<p>
|
|
|
|
The ideas proposed in this paper have been implemented in the open-source fmt
|
2016-08-20 14:39:10 +00:00
|
|
|
library. TODO: link and mention other implementations (Boost Format, FastFormat)
|
2016-08-19 16:33:59 +00:00
|
|
|
</p>
|
|
|
|
|
2016-08-19 23:43:10 +00:00
|
|
|
<h2><a name="References">References</a></h2>
|
|
|
|
|
2016-08-19 16:33:59 +00:00
|
|
|
<p>
|
|
|
|
<a name="1">[1]</a>
|
2016-08-24 14:41:07 +00:00
|
|
|
<cite>The <code>fprintf</code> function. ISO/IEC 9899:2011. 7.21.6.1.</cite><br>
|
2016-08-19 16:33:59 +00:00
|
|
|
<a name="2">[2]</a>
|
|
|
|
<cite><a href="http://pubs.opengroup.org/onlinepubs/009695399/functions/fprintf.html">
|
|
|
|
fprintf, printf, snprintf, sprintf - print formatted output</a>. The Open
|
2016-08-24 14:41:07 +00:00
|
|
|
Group Base Specifications Issue 6 IEEE Std 1003.1, 2004 Edition.</cite><br>
|
2016-08-19 16:33:59 +00:00
|
|
|
<a name="3">[3]</a>
|
|
|
|
<cite><a href="https://docs.python.org/3/library/string.html#format-string-syntax">
|
2016-08-24 14:41:07 +00:00
|
|
|
6.1.3. Format String Syntax</a>. Python 3.5.2 documentation.</cite><br>
|
2016-08-19 16:33:59 +00:00
|
|
|
<a name="4">[4]</a>
|
|
|
|
<cite><a href="https://msdn.microsoft.com/en-us/library/system.string.format(v=vs.110).aspx">
|
2016-08-24 14:41:07 +00:00
|
|
|
String.Format Method</a>. .NET Framework Class Library.</cite><br>
|
2016-08-19 16:33:59 +00:00
|
|
|
<a name="5">[5]</a>
|
|
|
|
<cite><a href="https://doc.rust-lang.org/std/fmt/">
|
2016-08-24 14:41:07 +00:00
|
|
|
Module <code>std::fmt</code></a>. The Rust Standard Library.</cite><br>
|
2016-08-19 23:43:10 +00:00
|
|
|
<a name="6">[6]</a>
|
|
|
|
<cite><a href="https://msdn.microsoft.com/en-us/library/56e442dc(v=vs.120).aspx">
|
|
|
|
Format Specification Syntax: printf and wprintf Functions</a>. C++ Language and
|
2016-08-24 14:41:07 +00:00
|
|
|
Standard Libraries.</cite><br>
|
2016-08-21 14:50:31 +00:00
|
|
|
<a name="7">[7]</a>
|
|
|
|
<cite><a href="ftp://ftp.gnu.org/old-gnu/Manuals/gawk-3.1.0/html_chapter/gawk_11.html">
|
2016-08-24 14:41:07 +00:00
|
|
|
10.4.2 Rearranging printf Arguments</a>. The GNU Awk User's Guide.</cite><br>
|
2016-08-19 16:33:59 +00:00
|
|
|
</p>
|
|
|
|
|
|
|
|
</body>
|