Module alloc::fmt 1.0.0[−][src]
Expand description
用于格式化和打印 Strings 的实用工具。
该模块包含对 format! 语法扩展的运行时支持。
该宏在编译器中实现,以发出对该模块的调用,以便在运行时将参数格式化为字符串。
Usage
format! 宏旨在使那些使用 C 的 printf/fprintf 函数或 Python 的 str.format 函数的用户熟悉。
format! 扩展的一些示例是:
format!("Hello"); // => "Hello"
format!("Hello, {}!", "world"); // => "Hello, world!"
format!("The number is {}", 1); // => "The number is 1"
format!("{:?}", (3, 4)); // => "(3, 4)"
format!("{value}", value=4); // => "4"
let people = "Rustaceans";
format!("Hello {people}!"); // => "Hello Rustaceans!"
format!("{} {}", 1, 2); // => "1 2"
format!("{:04}", 42); // => 带前导零的 "0042"
format!("{:#?}", (100, 200)); // => "(
// 100,
// 200, )"
//Run从这些中,您可以看到第一个参数是格式字符串。编译器要求它是字符串字面量; 它不能是传入的变量 (以执行有效性检查)。 然后,编译器将解析格式字符串,并确定所提供的参数列表是否适合传递给该格式字符串。
要将单个值转换为字符串,请使用 to_string 方法。这将使用 Display 格式 trait。
位置参数
每个格式化参数都可以指定它引用的值参数,如果省略,则假定它是 “下一个参数”。
例如,格式字符串 {} {} {} 将带有三个参数,并且将按照给定的顺序对其进行格式化。
但是,格式字符串 {2} {1} {0} 将以相反的顺序格式化参数。
一旦开始将两种类型的位置说明符混合在一起,事情就会变得有些棘手。可以将 “下一个参数” 说明符可以看作是参数的迭代器。 每次看到 “下一个参数” 说明符时,迭代器都会前进。这会导致这样的行为:
format!("{1} {} {0} {}", 1, 2); // => "2 1 1 2"Run看到第一个 {} 时,尚未对参数进行内部迭代,因此它将打印第一个参数。然后,在到达第二个 {} 时,迭代器已前进到第二个参数。
本质上,在位置说明符方面,明确命名其参数的参数不会影响未命名参数的参数。
必须使用格式字符串才能使用其所有参数,否则将导致编译时错误。您可能在格式字符串中多次引用同一参数。
命名参数
Rust 本身不具有类似于 Python 的等效于函数的命名参数,但是 format! 宏是一种语法扩展,允许它利用命名参数。
命名参数列在参数列表的末尾,并具有以下语法:
identifier '=' expression例如,以下 format! 表达式都使用命名参数:
format!("{argument}", argument = "test"); // => "test"
format!("{name} {}", 1, name = 2); // => "2 1"
format!("{a} {c} {b}", a="a", b='b', c=3); // => "a 3 b"Run如果命名参数没有出现在参数列表中,format! 将引用当前作用域中的同名变量。
let argument = 2 + 2;
format!("{argument}"); // => "4"
fn make_string(a: u32, b: &str) -> String {
format!("{b} {a}")
}
make_string(927, "label"); // => "label 927"Run在具有名称的参数之后放置位置参数 (那些没有名称的参数) 是无效的。与位置参数一样,提供格式字符串未使用的命名参数也是无效的。
格式化参数
每个要格式化的参数都可以通过许多格式化参数进行转换 (对应于 语法) 中的 format_spec。这些参数会影响所格式化内容的字符串表示形式。
Width
// 所有这些打印 "Hello x !"
println!("Hello {:5}!", "x");
println!("Hello {:1$}!", "x", 5);
println!("Hello {1:0$}!", 5, "x");
println!("Hello {:width$}!", "x", width = 5);
let width = 5;
println!("Hello {:width$}!", "x");Run这是格式应使用的 “最小宽度” 的参数。 如果值的字符串不能填满这么多字符,则 fill/alignment 指定的填充将用于占用所需的空间 (请参见下文)。
通过添加后缀 $ (表示第二个参数是指定宽度的 usize),也可以在参数列表中以 usize 的形式提供宽度值。
使用 Dollar 语法引用参数不会影响 “下一个参数” 计数器,因此按位置引用参数或使用命名参数通常是一个好主意。
Fill/Alignment
assert_eq!(format!("Hello {:<5}!", "x"), "Hello x !");
assert_eq!(format!("Hello {:-<5}!", "x"), "Hello x----!");
assert_eq!(format!("Hello {:^5}!", "x"), "Hello x !");
assert_eq!(format!("Hello {:>5}!", "x"), "Hello x!");Run可选的填充字符和对齐方式通常与 width 参数一起提供。必须在 width 之前,: 之后定义。
这表示如果要格式化的值小于 width,则将在其周围打印一些额外的字符。
对于不同的对齐方式,填充有以下变体:
[fill]<- 参数在width列中左对齐[fill]^- 参数在width列中居中对齐[fill]>- 参数在width列中右对齐
非数字的默认 fill/alignment 是空格,并且左对齐。数字格式器的默认值也是空格字符,但带有右对齐。
如果为数字指定了 0 标志 (见下文),则隐式填充字符为 0。
请注意,某些类型可能不会实现对齐。特别是,对于 Debug trait,通常不会实现该功能。
确保应用填充的一种好方法是格式化输入,然后填充此结果字符串以获得输出:
println!("Hello {:^15}!", format!("{:?}", Some("hi"))); // => "Hello Some("hi") !"RunSign/#/0
assert_eq!(format!("Hello {:+}!", 5), "Hello +5!");
assert_eq!(format!("{:#x}!", 27), "0x1b!");
assert_eq!(format!("Hello {:05}!", 5), "Hello 00005!");
assert_eq!(format!("Hello {:05}!", -5), "Hello -0005!");
assert_eq!(format!("{:#010x}!", 27), "0x0000001b!");Run这些都是更改格式化程序行为的标志。
+- 这适用于数字类型并指示应始终打印符号。默认情况下从不打印正号,默认情况下仅对有符号值打印负号。 该标志指示应始终打印正确的符号 (+或-)。-- 当前未使用#- 此标志表示应使用 “alternate” 打印形式。替代形式为:#?- 漂亮地打印Debug格式 (添加换行符和缩进)#x- 在参数前面加上0x#X- 在参数前面加上0x#b- 在参数前面加上0b#o- 在参数前面加上0o
0- 这用于指示对于整数格式,向width的填充应该使用0字符,并且是符号感知的。 像{:08}这样的格式将为整数1产生00000001,而相同格式将为整数-1产生-0000001。 请注意,负版本的零比正版本的少零。 请注意,填充零总是放在符号 (如果有) 之后和数字之前。当与#标志一起使用时,将应用类似的规则:在前缀之后但在数字之前插入填充零。 前缀包括在总宽度中。
Precision
对于非数字类型,可以将其视为 “最大宽度”。
如果结果字符串的长度大于此宽度,则将其截断为这么多个字符,并且如果设置了这些参数,则会使用适当的 fill,alignment 和 width 发出该截断的值。
对于整数类型,这将被忽略。
对于浮点类型,这指示小数点后应打印多少位。
有三种可能的方法来指定所需的 precision:
-
一个整数
.N:整数
N本身就是精度。 -
整数或名称后跟美元符号
.N$:使用格式参数
N(必须是usize) 作为精度。 -
星号
.*:.*意味着这个{...}与两个格式输入关联,而不是一个:第一个输入保存usize精度,第二个保存要打印的值。 请注意,在这种情况下,如果使用格式字符串{<arg>:<spec>.*},则<arg>部分将引用* value * 进行打印,并且precision必须位于<arg>之前的输入中。
例如,以下所有调用均打印相同的内容 Hello x is 0.01000:
// Hello {arg 0 ("x")} is {arg 1 (0.01) with precision specified inline (5)}
println!("Hello {0} is {1:.5}", "x", 0.01);
// Hello {arg 1 ("x")} is {arg 2 (0.01) with precision specified in arg 0 (5)}
println!("Hello {1} is {2:.0$}", 5, "x", 0.01);
// Hello {arg 0 ("x")} is {arg 2 (0.01) with precision specified in arg 1 (5)}
println!("Hello {0} is {2:.1$}", "x", 5, 0.01);
// Hello {next arg ("x")} is {second of next two args (0.01) with precision specified in first of next two args (5)}
//
println!("Hello {} is {:.*}", "x", 5, 0.01);
// Hello {next arg ("x")} is {arg 2 (0.01) with precision specified in its predecessor (5)}
//
println!("Hello {} is {2:.*}", "x", 5, 0.01);
// Hello {next arg ("x")} is {arg "number" (0.01) with precision specified in arg "prec" (5)}
//
println!("Hello {} is {number:.prec$}", "x", prec = 5, number = 0.01);Run而这些:
println!("{}, `{name:.*}` has 3 fractional digits", "Hello", 3, name=1234.56);
println!("{}, `{name:.*}` has 3 characters", "Hello", 3, name="1234.56");
println!("{}, `{name:>8.*}` has 3 right-aligned characters", "Hello", 3, name="1234.56");Run打印三个明显不同的内容:
Hello, `1234.560` has 3 fractional digits
Hello, `123` has 3 characters
Hello, ` 123` has 3 right-aligned charactersLocalization
在某些编程语言中,字符串格式函数的行为取决于操作系统的语言环境设置。 Rust 标准库提供的格式函数没有任何语言环境的概念,并且无论用户配置如何,在所有系统上都会产生相同的结果。
例如,即使系统区域设置使用小数点分隔符 (而不是点),以下代码也将始终打印 1.5。
println!("The value is {}", 1.5);RunEscaping
字面量字符 { 和 } 可以通过在它们之前添加相同的字符而包含在字符串中。例如,{ 字符使用 {{ 进行转义,而 } 字符使用 }} 进行转义。
assert_eq!(format!("Hello {{}}"), "Hello {}");
assert_eq!(format!("{{ Hello"), "{ Hello");RunSyntax
总结一下,您可以在这里找到格式字符串的完整语法。
所用格式语言的语法是从其他语言中提取的,因此不应太陌生。参数使用类似 Python 的语法格式化,这意味着参数被 {} 包围,而不是类似 C 的 %。
格式化语法的实际语法为:
format_string := text [ maybe_format text ] *
maybe_format := '{' '{' | '}' '}' | format
format := '{' [ argument ] [ ':' format_spec ] '}'
argument := integer | identifier
format_spec := [[fill]align][sign]['#']['0'][width]['.' precision]type
fill := character
align := '<' | '^' | '>'
sign := '+' | '-'
width := count
precision := count | '*'
type := '' | '?' | 'x?' | 'X?' | identifier
count := parameter | integer
parameter := argument '$'在上述语法中,text 不得包含任何 '{' 或 '}' 字符。
格式化 traits
当请求使用特定类型的参数格式化时,实际上是在请求将参数归因于特定的 trait。
这允许通过 {:x} 格式化多种实际类型 (例如 i8 和 isize)。类型到 traits 的当前映射是:
- nothing ⇒
Display ?⇒Debugx?⇒Debug带有小写十六进制整数X?⇒Debug带有大写十六进制整数o⇒Octalx⇒LowerHexX⇒UpperHexp⇒Pointerb⇒Binarye⇒LowerExpE⇒UpperExp
这意味着可以使用 {:b} 格式化实现 fmt::Binary trait 的任何类型的参数。标准库还为许多原始类型提供了针对这些 traits 的实现。
如果未指定格式 (如 {} 或 {:6}),则使用的格式 trait 为 Display trait。
当为您自己的类型实现格式 trait 时,您将必须实现签名的方法:
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {Run您的类型将作为 self by - 引用传递,然后函数应将输出发出到 f.buf 流中。正确遵守所请求的格式设置参数,取决于每种格式 trait 的实现。
这些参数的值将在 Formatter 结构体的字段中列出。为了解决这个问题,Formatter 结构体还提供了一些辅助方法。
此外,该函数的返回值是 fmt::Result,它是 Result<(), std::fmt::Error> 的类型别名。
格式化实现应确保它们传播来自 Formatter 的错误 (例如,调用 write! 时)。
但是,它们绝不能虚假地返回错误。
即,格式化实现必须并且仅在传入的 Formatter 返回错误的情况下才返回错误。
这是因为,与函数签名可能暗示的相反,字符串格式是一项可靠的操作。
该函数仅返回结果,因为写入底层流可能会失败,并且它必须提供一种方法来将已发生错误的事实传播回栈。
实现格式 traits 的示例如下所示:
use std::fmt;
#[derive(Debug)]
struct Vector2D {
x: isize,
y: isize,
}
impl fmt::Display for Vector2D {
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
// `f` 值实现 `Write` trait,这就是 `write`! 宏正在等待。
// 请注意,这种格式化将忽略为格式化字符串而提供的各种标志。
//
write!(f, "({}, {})", self.x, self.y)
}
}
// 不同的 traits 允许类型的不同形式的输出。
// 此格式的含义是打印 vector 的大小。
impl fmt::Binary for Vector2D {
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
let magnitude = (self.x * self.x + self.y * self.y) as f64;
let magnitude = magnitude.sqrt();
// 通过使用 Formatter 对象上的帮助器方法 `pad_integral`,尊重格式设置标志。
// 有关详细信息,请参见方法文档,并且函数 `pad` 可用于填充字符串。
//
//
let decimals = f.precision().unwrap_or(3);
let string = format!("{:.*}", decimals, magnitude);
f.pad_integral(true, "", &string)
}
}
fn main() {
let myvector = Vector2D { x: 3, y: 4 };
println!("{}", myvector); // => "(3, 4)"
println!("{:?}", myvector); // => "Vector2D {x: 3, y:4}"
println!("{:10.3b}", myvector); // => " 5.000"
}Runfmt::Display 与 fmt::Debug
这两种格式 traits 具有不同的用途:
fmt::Display实现断言该类型可以始终忠实地表示为 UTF-8 字符串。并非所有类型都实现Displaytrait。fmt::Debug实现应该为所有公共类型实现。 输出通常会尽可能忠实地代表内部状态。Debugtrait 的目的是方便调试 Rust 代码。在大多数情况下,建议使用#[derive(Debug)]就足够了。
这两个 traits 的输出的一些例子:
assert_eq!(format!("{} {:?}", 3, 4), "3 4");
assert_eq!(format!("{} {:?}", 'a', 'b'), "a 'b'");
assert_eq!(format!("{} {:?}", "foo\n", "bar\n"), "foo\n \"bar\\n\"");Run相关宏
format! 系列中有许多相关的宏。当前实现的是:
format! // 如上所述
write! // 第一个参数是 &mut io::Write,目的地
writeln! // 与 write 相同,但追加了一个换行符
print! // 格式字符串被打印到标准输出
println! // 与 print 相同,但追加了一个换行符
eprint! // 格式字符串被打印到标准错误
eprintln! // 与 eprint 相同,但追加了一个换行符
format_args! // 如下面所描述的。Runwrite!
这和 writeln! 是两个宏,用于将格式字符串发射到指定的流。这用于防止格式字符串的中间分配,而是直接写入输出。
在后台,此函数实际上是在 std::io::Write trait 上定义的 write_fmt 函数。
示例用法是:
use std::io::Write;
let mut w = Vec::new();
write!(&mut w, "Hello {}!", "world");Runprint!
此和 println! 将其输出发送到 stdout。与 write! 宏类似,这些宏的目标是避免在打印输出时进行中间分配。示例用法是:
print!("Hello {}!", "world");
println!("I have a newline {}", "character at the end");Runeprint!
eprint! 和 eprintln! 宏分别与 print! 和 println! 相同,只不过它们将其输出发送到 stderr。
format_args!
这是一个奇怪的宏,用于安全地传递描述格式字符串的不透明对象。该对象不需要创建任何堆分配,并且仅引用栈上的信息。 在幕后,所有相关的宏都在此方面实现。 首先,一些示例用法是:
use std::fmt;
use std::io::{self, Write};
let mut some_writer = io::stdout();
write!(&mut some_writer, "{}", format_args!("print with a {}", "macro"));
fn my_fmt_fn(args: fmt::Arguments) {
write!(&mut io::stdout(), "{}", args);
}
my_fmt_fn(format_args!(", or a {} too", "function"));Runformat_args! 宏的结果是 fmt::Arguments 类型的值。
然后可以将此结构体传递到此模块内部的 write 和 format 函数,以处理格式字符串。
该宏的目的是在处理格式化字符串时甚至进一步防止中间分配。
例如,日志记录库可以使用标准格式语法,但是它将在内部绕过此结构体,直到确定了输出应该到达的位置为止。
Macros
派生宏,生成 Debug trait 的 impl。
Structs
该结构体表示格式字符串及其参数的安全预编译版本。 由于无法安全地完成此操作,因此无法在运行时生成该文件,因此未提供任何构造函数,并且该字段为私有字段以防止修改。
一个有助于 fmt::Debug 实现的结构体。
一个有助于 fmt::Debug 实现的结构体。
一个有助于 fmt::Debug 实现的结构体。
一个有助于 fmt::Debug 实现的结构体。
一个有助于 fmt::Debug 实现的结构体。
将消息格式化为流后返回的错误类型。
格式化配置。
Enums
Formatter::align 返回的可能的对齐方式
Traits
b 格式。
? 格式。
空格式的格式 trait,{}。
e 格式。
x 格式。
o 格式。
p 格式。
E 格式。
X 格式。
一个用于写入或格式化为 Unicode 接受的缓冲区或流的 trait。
Functions
write 函数接受一个输出流,以及一个可以与 format_args! 宏预编译的 Arguments 结构体。
Type Definitions
格式化程序方法返回的类型。