Auto merge of #85391 - Mark-Simulacrum:opt-tostring, r=scottmcm
Avoid zero-length memcpy in formatting This has two separate and somewhat orthogonal commits. The first change adjusts the ToString general impl for all types that implement Display; it no longer uses the full format machinery, rather directly falling onto a `std::fmt::Display::fmt` call. The second change directly adjusts the general core::fmt::write function which handles the production of format_args! to avoid zero-length push_str calls. Both changes target the fact that push_str will still call memmove internally (or a similar function), as it doesn't know the length of the passed string. For zero-length strings in particular, this is quite expensive, and even for very short (several bytes long) strings, this is also expensive. Future work in this area may wish to have us fallback to write_char or similar, which may be cheaper on the (typically) short strings between the interpolated pieces in format_args!.
This commit is contained in:
commit
a426fc37f2
@ -2323,9 +2323,10 @@ impl<T: fmt::Display + ?Sized> ToString for T {
|
||||
// to try to remove it.
|
||||
#[inline]
|
||||
default fn to_string(&self) -> String {
|
||||
use fmt::Write;
|
||||
let mut buf = String::new();
|
||||
buf.write_fmt(format_args!("{}", self))
|
||||
let mut formatter = core::fmt::Formatter::new(&mut buf);
|
||||
// Bypass format_args!() to avoid write_str with zero-length strs
|
||||
fmt::Display::fmt(self, &mut formatter)
|
||||
.expect("a Display implementation returned an error unexpectedly");
|
||||
buf
|
||||
}
|
||||
|
@ -221,6 +221,28 @@ pub struct Formatter<'a> {
|
||||
buf: &'a mut (dyn Write + 'a),
|
||||
}
|
||||
|
||||
impl<'a> Formatter<'a> {
|
||||
/// Creates a new formatter with default settings.
|
||||
///
|
||||
/// This can be used as a micro-optimization in cases where a full `Arguments`
|
||||
/// structure (as created by `format_args!`) is not necessary; `Arguments`
|
||||
/// is a little more expensive to use in simple formatting scenarios.
|
||||
///
|
||||
/// Currently not intended for use outside of the standard library.
|
||||
#[unstable(feature = "fmt_internals", reason = "internal to standard library", issue = "none")]
|
||||
#[doc(hidden)]
|
||||
pub fn new(buf: &'a mut (dyn Write + 'a)) -> Formatter<'a> {
|
||||
Formatter {
|
||||
flags: 0,
|
||||
fill: ' ',
|
||||
align: rt::v1::Alignment::Unknown,
|
||||
width: None,
|
||||
precision: None,
|
||||
buf,
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// NB. Argument is essentially an optimized partially applied formatting function,
|
||||
// equivalent to `exists T.(&T, fn(&T, &mut Formatter<'_>) -> Result`.
|
||||
|
||||
@ -1075,22 +1097,16 @@ pub trait UpperExp {
|
||||
/// [`write!`]: crate::write!
|
||||
#[stable(feature = "rust1", since = "1.0.0")]
|
||||
pub fn write(output: &mut dyn Write, args: Arguments<'_>) -> Result {
|
||||
let mut formatter = Formatter {
|
||||
flags: 0,
|
||||
width: None,
|
||||
precision: None,
|
||||
buf: output,
|
||||
align: rt::v1::Alignment::Unknown,
|
||||
fill: ' ',
|
||||
};
|
||||
|
||||
let mut formatter = Formatter::new(output);
|
||||
let mut idx = 0;
|
||||
|
||||
match args.fmt {
|
||||
None => {
|
||||
// We can use default formatting parameters for all arguments.
|
||||
for (arg, piece) in iter::zip(args.args, args.pieces) {
|
||||
formatter.buf.write_str(*piece)?;
|
||||
if !piece.is_empty() {
|
||||
formatter.buf.write_str(*piece)?;
|
||||
}
|
||||
(arg.formatter)(arg.value, &mut formatter)?;
|
||||
idx += 1;
|
||||
}
|
||||
@ -1099,7 +1115,9 @@ pub fn write(output: &mut dyn Write, args: Arguments<'_>) -> Result {
|
||||
// Every spec has a corresponding argument that is preceded by
|
||||
// a string piece.
|
||||
for (arg, piece) in iter::zip(fmt, args.pieces) {
|
||||
formatter.buf.write_str(*piece)?;
|
||||
if !piece.is_empty() {
|
||||
formatter.buf.write_str(*piece)?;
|
||||
}
|
||||
// SAFETY: arg and args.args come from the same Arguments,
|
||||
// which guarantees the indexes are always within bounds.
|
||||
unsafe { run(&mut formatter, arg, &args.args) }?;
|
||||
|
Loading…
x
Reference in New Issue
Block a user