-
Notifications
You must be signed in to change notification settings - Fork 13.2k
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Debug trait for tuples, array/slices, Vec, String (etc?) do not respect width
parameter
#30164
Comments
, which is kind of ironic - a trait named |
As a workaround for now I've been using
which works fine. However, would be great if this would work natively |
I came here to file a similar issue with alignment specifiers. But it's likely the same bug. Example: println!("|{:^11?}|", 3); // prints: | 3 |
println!("|{:^11?}|", "hi"); // prints: |"hi"| |
I'm not sure if there have been solutions proposed for this anywhere. I'm mostly thinking of Notes:
Bits of a solution?
|
I just got annoyed for 20 minutes or so that |
I'm not sure if this is related, but I seem to encounter a similar problem when using custom use std::fmt;
struct Foo {
bar: i32,
}
impl fmt::Display for Foo {
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
write!(f, "{:f>5}", self.bar)
}
}
fn main() {
println!("|{:<30}|", "");
println!("|{:<30}|", 12);
println!("|{:<30}|", Foo {bar: 12});
println!("|{:<30}|", format!("{}", Foo {bar: 12}));
} Output:
In this case the aforementioned workaround to use Edit: I realize now, this is because my impl fmt::Display for Foo {
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
f.pad(&format!("{:f>5}", self.bar))
}
} With this, the expected result is printed. Hopefully this helps anyone else trying to fix their |
This issue has been reported independently several times over the years: see issues #43909, #55584, #55749, #83361, #87905, #88059. Clearly people do try to apply formatting parameters to I tested some of the most common types for how their
To clarify the last 2: Slices, I will try to implement the following changes:
|
Make `Duration` respect `width` when formatting using `Debug` When printing or writing a `std::time::Duration` using `Debug` formatting, it previously completely ignored any specified `width`. This is unlike types like integers and floats, which do pad to `width`, for both `Display` and `Debug`, though not all types consider `width` in their `Debug` output (see e.g. rust-lang#30164). Curiously, `Duration`'s `Debug` formatting *did* consider `precision`. This PR makes `Duration` pad to `width` just like integers and floats, so that ```rust format!("|{:8?}|", Duration::from_millis(1234)) ``` returns ``` |1.234s | ``` Before you ask "who formats `Debug` output?", note that `Duration` doesn't actually implement `Display`, so `Debug` is currently the only way to format `Duration`s. I think that's wrong, and `Duration` should get a `Display` implementation, but in the meantime there's no harm in making the `Debug` formatting respect `width` rather than ignore it. I chose the default alignment to be left-aligned. The general rule Rust uses is: numeric types are right-aligned by default, non-numeric types left-aligned. It wasn't clear to me whether `Duration` is a numeric type or not. The fact that a formatted `Duration` can end with suffixes of variable length (`"s"`, `"ms"`, `"µs"`, etc.) made me lean towards left-alignment, but it would be trivial to change it. Fixes issue rust-lang#88059.
Make `Duration` respect `width` when formatting using `Debug` When printing or writing a `std::time::Duration` using `Debug` formatting, it previously completely ignored any specified `width`. This is unlike types like integers and floats, which do pad to `width`, for both `Display` and `Debug`, though not all types consider `width` in their `Debug` output (see e.g. rust-lang#30164). Curiously, `Duration`'s `Debug` formatting *did* consider `precision`. This PR makes `Duration` pad to `width` just like integers and floats, so that ```rust format!("|{:8?}|", Duration::from_millis(1234)) ``` returns ``` |1.234s | ``` Before you ask "who formats `Debug` output?", note that `Duration` doesn't actually implement `Display`, so `Debug` is currently the only way to format `Duration`s. I think that's wrong, and `Duration` should get a `Display` implementation, but in the meantime there's no harm in making the `Debug` formatting respect `width` rather than ignore it. I chose the default alignment to be left-aligned. The general rule Rust uses is: numeric types are right-aligned by default, non-numeric types left-aligned. It wasn't clear to me whether `Duration` is a numeric type or not. The fact that a formatted `Duration` can end with suffixes of variable length (`"s"`, `"ms"`, `"µs"`, etc.) made me lean towards left-alignment, but it would be trivial to change it. Fixes issue rust-lang#88059.
Consider the following println:
This is using the
width
parameter to ensure that the output occupies at leastwidth
characters. (One can provide other arguments like fill/alignment to adjust the fill character or whether the output is left/right/center- alligned.) See docs here: https://doc.rust-lang.org/std/fmt/#widthIn other words, it prints this (note the distance between the two occurrences of
|
):So, the problem:
The
Debug
trait seems to honorwidth
for numeric and pointer types, but not for other types.For example:
does not necessarily put at least 30 characters between the two
|
characters that it prints; in the case ofmsg
given above, it prints:Here is a playpen illustrating the problem on several inputs.
The text was updated successfully, but these errors were encountered: