Install
Install Rust
linux
$ curl --proto '=https' --tlsv1.2 https://sh.rustup.rs -sSf | sh
windows
download rustup_init:
https://static.rust-lang.org/rustup/dist/x86_64-pc-windows-msvc/rustup-init.exe
more see rustup doc
https://rust-lang.github.io/rustup/
more
More rust doc book-installation
update rust
rustup update
delete rustup
rustup self uninstall
'when error'
git clone https://github.com/rust-lang/rustup
cargo build --release
mv ./target/release/rustup-init.exe rustup
...
rustup
rustup show
rustup check
rustup doc
rustup target list
cargo
cargo new project_name
cargo new --lib project_name
more
rustup doc --cargo
Then to start using Rust.
rustc
cargo
rustup
More rust doc book-installation
First
First learning computer language
Printf
cargo new two
cd ./two/src
main.rs
fn main() {
println!("Hello, world!");
}
Then
Cargo expand
#![feature(prelude_import)]
#[prelude_import]
use std::prelude::v1::*;
#[macro_use]
extern crate std;
fn main() {
{
::std::io::_print(::core::fmt::Arguments::new_v1(
&["Hello, world!\n"],
&match () {
() => [],
},
));
};
}
see rustdoc doc/rust/html/std/fmt/index.html
+
- This is intended for numeric types and indicates that the sign should always be printed. Positive signs are never printed by default, and the negative sign is only printed by default for theSigned
trait. This flag indicates that the correct sign (+
or-
) should always be printed.-
- Currently not used#
- This flag indicates that the "alternate" form of printing should be used. The alternate forms are:#?
- pretty-print theDebug
formatting#x
- precedes the argument with a0x
#X
- precedes the argument with a0x
#b
- precedes the argument with a0b
#o
- precedes the argument with a0o
0
- This is used to indicate for integer formats that the padding towidth
should both be done with a0
character as well as be sign-aware. A format like{:08}
would yield00000001
for the integer1
, while the same format would yield-0000001
for the integer-1
. Notice that the negative version has one fewer zero than the positive version. Note that padding zeros are always placed after the sign (if any) and before the digits. When used together with the#
flag, a similar rule applies: padding zeros are inserted after the prefix but before the digits. The prefix is included in the total width.
pointer
fn main() {
let x = &42;
let address = format!("{:p}", x); // this produces something like '0x7f06092ac6d0'
print!("{}", address);
}
cargo expand
#![feature(prelude_import)]
#[prelude_import]
use std::prelude::v1::*;
#[macro_use]
extern crate std;
fn main() {
let x = &42;
let address = {
let res = ::alloc::fmt::format(::core::fmt::Arguments::new_v1(
&[""],
&match (&x,) {
(arg0,) => [::core::fmt::ArgumentV1::new(
arg0,
::core::fmt::Pointer::fmt,
)],
},
));
res
};
::std::io::_print(::core::fmt::Arguments::new_v1(
&[""],
&match (&address,) {
(arg0,) => [::core::fmt::ArgumentV1::new(
arg0,
::core::fmt::Display::fmt,
)],
},
));
}
so:
fn main() {
let x = &42;
print!("{:p}",x);
}
cargo expand
#![feature(prelude_import)]
#[prelude_import]
use std::prelude::v1::*;
#[macro_use]
extern crate std;
fn main() {
let x = &42;
::std::io::_print(::core::fmt::Arguments::new_v1(
&[""],
&match (&x,) {
(arg0,) => [::core::fmt::ArgumentV1::new(
arg0,
::core::fmt::Pointer::fmt,
)],
},
));
}
These are all flags altering the behavior of the formatter.
+
- This is intended for numeric types and indicates that the sign should always be printed. Positive signs are never printed by default, and the negative sign is only printed by default for theSigned
trait. This flag indicates that the correct sign (+
or-
) should always be printed.-
- Currently not used#
- This flag indicates that the "alternate" form of printing should be used. The alternate forms are:#?
- pretty-print theDebug
formatting#x
- precedes the argument with a0x
#X
- precedes the argument with a0x
#b
- precedes the argument with a0b
#o
- precedes the argument with a0o
0
- This is used to indicate for integer formats that the padding towidth
should both be done with a0
character as well as be sign-aware. A format like{:08}
would yield00000001
for the integer1
, while the same format would yield-0000001
for the integer-1
. Notice that the negative version has one fewer zero than the positive version. Note that padding zeros are always placed after the sign (if any) and before the digits. When used together with the#
flag, a similar rule applies: padding zeros are inserted after the prefix but before the digits. The prefix is included in the total width.
more
macro
using
just only
- Patterns
- Statements
- Expressions
- Items
impl
Items
It is ban for list
- Identifiers
- Match arms
- Struct fields
- Types
First
macro_rules!
macro_rules! $name{
...
}
macro_rules! for {
()=>{1+3};
}
//four!()、four![] 或者 four!{}
variable
item
: an item, like a function, struct, module, etc.block
: a block (i.e. a block of statements and/or an expression, surrounded by braces)stmt
: a statementpat
: a patternexpr
: an expressionty
: a typeident
: an identifierpath
: a path (e.g.foo
,::std::mem::replace
,transmute::<_, int>
, …)meta
: a meta item; the things that go inside#[...]
and#![...]
attributestt
: a single token tree
examples:
macro_rules! one_expression {
($e:expr) => {...};
}
macro_rules! times_five {
($e:expr) => {5 * $e};
}
macro_rules! multiply_add {
($a:expr, $b:expr, $c:expr) => {$a * ($b + $c)};
}
$
is a literal dollar token.( ... )
is the paren-grouped pattern being repeated.sep
is an optional separator token. Common examples are,
, and;
.rep
is the required repeat control. Currently, this can be either
*
(indicating zero or more repeats) or
+
(indicating one or more repeats). You cannot write "zero or one" or any other more specific counts or ranges.
macro_rules! vec_strs {
(
// Start a repetition:
$(
// Each repeat must contain an expression...
$element:expr
)
// ...separated by commas...
,
// ...zero or more times.
*
) => {
// Enclose the expansion in a block so that we can use
// multiple statements.
{
let mut v = Vec::new();
// Start a repetition:
$(
// Each repeat will contain the following statement, with
// $element replaced with the corresponding expression.
v.push(format!("{}", $element));
)*
v
}
};
}
more
Type
number
flot
use num_traits::float::Float; // 0.2.6
use std::ops::Mul;
fn scale_float<T>(x: T) -> T
where
T: Float + Mul<f64, Output = T>,
{
x * 0.54
}
fn main() {
let a: f64 = scale_float(1.23);
}
integer
fn f<T: FromU64>(v: u64) -> T {
FromU64::from_u64(T)
}
Variables without values (//easy_rust)
A variable without a value is called an "uninitialized" variable. Uninitialized means "hasn't started yet". They are simple: just write let
and the variable name:
fn main() { let my_variable; // ⚠️ }
But you can't use it yet, and Rust won't compile if anything is uninitialized.
But sometimes they can be useful. A good example is when:
- You have a code block and the value for your variable is inside it, and
- The variable needs to live outside of the code block.
fn loop_then_return(mut counter: i32) -> i32 { loop { counter += 1; if counter % 50 == 0 { break; } } counter } fn main() { let my_number; { // Pretend we need to have this code block let number = { // Pretend there is code here to make a number // Lots of code, and finally: 57 }; my_number = loop_then_return(number); } println!("{}", my_number); }
This prints 100
.
You can see that my_number
was declared in the main()
function, so it lives until the end. But it gets its value from inside a loop. However, that value lives as long as my_number
, because my_number
has the value. And if you wrote let my_number = loop_then_return(number)
inside the block, it would just die right away.
It helps to imagine if you simplify the code. loop_then_return(number)
gives the result 100, so let's delete it and write 100
instead. Also, now we don't need number
so we will delete it too. Now it looks like this:
fn main() { let my_number; { my_number = 100; } println!("{}", my_number); }
So it's almost like saying let my_number = { 100 };
.
Also note that my_number
is not mut
. We didn't give it a value until we gave it 50, so it never changed its value. In the end, the real code for my_number
is just let my_number = 100;
.
emblem
-
.
The function of struct witch parameter must be &self
-
::
-
;
-
,
any
#![allow(unused)]
#![feature(type_name_of_val)]
fn main() {
use std::any::type_name_of_val;
let x = 1;
println!("{}", type_name_of_val(&x));
let y = 1.0;
println!("{}", type_name_of_val(&y));
}
ref
#![allow(unused)]
fn main() {
let maybe_name = Some(String::from("Alice"));
// Using `ref`, the value is borrowed, not moved ...
match maybe_name {
Some(ref n) => println!("Hello, {}", n),
_ => println!("Hello, world"),
}
// ... so it's available here!
println!("Hello again, {}", maybe_name.unwrap_or("world".into()));
}
lifetime
let mut x = 5;
// let y = &x;
let y = &mut x;
println!("{}", x);
println!("{}", y);
/*error. let y = &mut x;
| ------ mutable borrow occurs here
7 | println!("{}", x);
| ^ immutable borrow occurs here
8 |
9 | println!("{}", y);
| - mutable borrow later used here
*/
variable
Functions
String
string-str
when str and string share in ...
AsRef
fn is_hello<T: AsRef<str>>(s: T) {
assert_eq!("hello", s.as_ref());
}
let s = "hello";
is_hello(s);
let s = "hello".to_string();
is_hello(s);
#![allow(unused)]
fn main() {
let s = String::from("hello");
let b = s.into_boxed_str();
println!("{}",b);
}
pub fn capitalize_first(input: &str) -> String {
let mut c = input.chars();
match c.next() {
None => String::new(),
Some(first) => first.to_uppercase().collect::<String>() + c.as_str(),
}
}
//Selcect number form string, it will be a usize.
String:String = ss.chars().filter(|x| x.is_digit(10)).collect();
//Selcect number form string, it will be a float.
String:String = ss.chars().filter(|x| x.is_digit(10)||x='.').collect();
struct
trait Shape {
fn area(&self)->f64;
}
trait Round {
fn get_radius(&self)->f64;
}
struct Circle {
radius: f64,
}
impl Round for Circle {
fn get_radius(&self) -> f64 { self.radius }
}
// 注意这里是 impl Trait for Trait
impl Shape for dyn Round {
fn area(&self) -> f64 {
std::f64::consts::PI * self.get_radius() * self.get_radius()
}
}
fn main() {
let c = Circle { radius : 2f64};
// 编译错误
//c.area();
let b = Box::new(Circle {radius : 4f64}) as Box<dyn Round>;
// 编译正确
let c = b.area();
println!("{}",c);
}
srqc-p94
// . and ::
struct T(i32);
impl T {
// 这是一个静态方法
fn func(this: &Self) {
println!{"value {}", this.0};
}
// 这不是一个静态方法
fn first(&Self) {
println!{"value {}", self.0};
}
}
fn main() {
let x = T(42);
// x.func(); 小数点方式调用是不合法的
T::func(&x);
let x = T(4);
x.first();
}
trait
impl trait
pub trait animal{
fn print_name(&self);
}
struct cat{
name:String
}
struct dog{
name:String
}
impl animal for cat{
fn print_name(&self){
println!("{}",self.name);
}
}
impl animal for dog{
fn print_name(&self){
println!("{}",self.name);
}
}
fn who(who:i32)->impl animal{ //注意只能返回同一个类型
if who== 1{
cat{name:"cat one ".to_string()}
} else{
cat{name:"cat two ".to_string()}
}
}
fn main(){
let a = who(1);
a.print_name();
}
struct Closure<F> {
data: (u8, u16),
func: F
}
impl<F> Closure<F>
where F: Fn(&(u8, u16)) -> &u8,
{
fn call(&self) -> &u8 {
(self.func)(&self.data)
}
}
fn do_it(data: &(u8, u16)) -> &u8 { &data.0 }
fn main() {
let clo = Closure{ data: (0, 1), func: do_it };
println!("{}", clo.call());
}
collections
hash
use std::collections::HashMap;
use std::hash::{Hash, Hasher};
macro_rules! seq_to_map {
($seq:expr) => {{
let mut map = HashMap::<_, _, RandomState>::default();
let mut iter = $seq.into_iter();
while let Some(e) = iter.next() {
map.entry(e).or_insert(iter.next().unwrap_or_default());
}
map
}};
}
fn seq_to_map<T: IntoIterator>(vec: T) -> HashMap<T::Item, T::Item>
where
T::Item: Hash + Default + Eq,
{
let mut map = HashMap::default();
let mut iter = vec.into_iter();
while let Some(e) = iter.next() {
map.entry(e).or_insert(iter.next().unwrap_or_default());
}
map
}
fn main() {
let mut v=Vec::new();
for i in "a1b2c3d4e5f6g7h8i9j".chars(){
v.push(i)
}
let h=seq_to_map(v);
println!("{:?}",h);
//vec.chunks_exact(2).map(|chunk| (chunk[0], chunk[1])).collect::<HashMap::<_, _>>()
}
use std::collections::hash_map::DefaultHasher;
use std::hash::{Hash, Hasher};
#[derive(Hash)]
struct Person {
id: u32,
name: String,
phone: u64,
}
let person1 = Person {
id: 5,
name: "Janet".to_string(),
phone: 555_666_7777,
};
let person2 = Person {
id: 5,
name: "Bob".to_string(),
phone: 555_666_7777,
};
assert!(calculate_hash(&person1) != calculate_hash(&person2));
fn calculate_hash<T: Hash>(t: &T) -> u64 {
let mut s = DefaultHasher::new();
t.hash(&mut s);
s.finish()
}
use std::collections::hash_map::DefaultHasher;
use std::hash::{Hash, Hasher};
struct Person {
id: u32,
name: String,
phone: u64,
}
impl Hash for Person {
fn hash<H: Hasher>(&self, state: &mut H) {
self.id.hash(state);
self.phone.hash(state);
}
}
let person1 = Person {
id: 5,
name: "Janet".to_string(),
phone: 555_666_7777,
};
let person2 = Person {
id: 5,
name: "Bob".to_string(),
phone: 555_666_7777,
};
assert_eq!(calculate_hash(&person1), calculate_hash(&person2));
fn calculate_hash<T: Hash>(t: &T) -> u64 {
let mut s = DefaultHasher::new();
t.hash(&mut s);
s.finish()
}
slice
copy_from_slice:
#[repr(C)]
pub struct Row {
id: i32,
user: [char; 32],
email: [char; 255],
}
impl Row {
pub fn new(id: i32, user_str: &str, email_str: &str) -> Self {
let mut user: [char; 32] = ['\0'; 32];
let mut email: [char; 255] = ['\0'; 255];
user.copy_from_slice(&user_str.chars().collect::<Box<[char]>>());
email.copy_from_slice(&email_str.chars().collect::<Box<[char]>>());
Row { id, user, email }
}
}
*** How about RandomState with new?***
#![allow(unused)]
fn main() {
use std::collections::HashMap;
use std::collections::hash_map::RandomState;
let s = RandomState::new();
let mut map = HashMap::with_hasher(s);
if map.is_empty(){
println!("The map it empty");
} else {
println!("{:?}",map);
}
map.insert(1, 2);
println!("{:?}",map)
}
Owner
trait
trait Foo {
fn run(&mut self) {
println!("Foo::run()");
}
}
fn run_base(foo: &mut Foo) {
println!("run_base()");
}
struct Bar;
impl Foo for Bar {
fn run(&mut self) {
run_base(self);
println!("Bar::run()");
}
}
fn main() {
let mut bar = Bar;
bar.run();
}
fn sum<T:From<f32>>(min:i32, max:i32)->T{
let result = 3.14 + min as f32 + max as f32;
return result.into();
}
fn main(){
let a:f64 = sum(15,100);
let b:f32 = sum(15,100);
let c = sum::<f32>(15,100);
let d = sum::<f64>(15,100);
println!("{},{},{},{}",a,b,c,d);
}
PhantomData
base
where
use std::fmt::Debug; #[derive(Debug)] struct Animal { name: String, age: u8, } fn print_item<T: Debug>(item: T) { println!("Here is your item: {:?}", item); } fn main() { let charlie = Animal { name: "Charlie".to_string(), age: 1, }; let number = 55; print_item(charlie); print_item(number); }
use std::fmt::Display; use std::cmp::PartialOrd; fn compare_and_display<T: Display, U: Display + PartialOrd>(statement: T, num_1: U, num_2: U) { println!("{}! Is {} greater than {}? {}", statement, num_1, num_2, num_1 > num_2); } fn main() { compare_and_display("Listen up!", 9, 8); }
use std::cmp::PartialOrd; use std::fmt::Display; fn compare_and_display<T, U>(statement: T, num_1: U, num_2: U) where T: Display, U: Display + PartialOrd, { println!("{}! Is {} greater than {}? {}", statement, num_1, num_2, num_1 > num_2); } fn main() { compare_and_display("Listen up!", 9, 8); }
Using where
is a good idea when you have many generic types.
Also note:
- If you have one type T and another type T, they must be the same.
- If you have one type T and another type U, they can be different. But they can also be the same.
Memory model
Lifetime
Begin
start from a simple function
fn main() {
let mut s:String="Hello world!".to_string();
println!("{}",get_first_word(&mut s));
}
/*
//It is Ok!
fn get_first_word(s:String)->String{
let ss:String=s.chars().take_while(|x| x.is_ascii_alphabetic()).clone().collect();
ss
}
*/
/*
// This is error.
// &ss cannot use fnnciton out.
fn fn get_first_word(s:String)->&str{
let ss:String=s.chars().take_while(|x| x.is_ascii_alphabetic()).clone().collect();
&ss
}
*/
// so, changed main string to mut and using follow function.
fn get_first_word(s:&mut String)->&str{
let key=s.find(' ');
match key{
Some(i)=> return &s[..i],
None => return &s[..],
}
&s
}
It is ok:
//https://github.com/GuilhermeVBeira/Rust-book-exercises/blob/master/4.%20Understanding%20Ownership/the_slice_type/src/main.rs
fn main() {
let my_string = String::from("hello world");
let word = first_word(&my_string[..]);
let my_string_literal = "hello world";
let word = first_word(&my_string_literal[..]);
let word = first_word(my_string_literal);
let a = [1, 2, 3, 4, 5];
println!("{:?}", a);
let slice = &a[1..3];
println!("{:?}", slice);
}
fn first_word(s: &str) -> &str {
let bytes = s.as_bytes();
for (i, &item) in bytes.iter().enumerate() {
if item == b' ' {
return &s[0..i];
}
}
&s[..]
}
where is moved lifetime
fn fn1(s: &mut String) {}
fn main() {
let mut s = "".to_string();
assert!(s.is_empty());
let s1 = &mut s;
fn1(s1); // s1 没有失效
s1.len();
{
let s2: &mut String = s1;
}
s1.len(); // s1 没有失效
{
let s2=s1;
//let s2: &mut String = s1;
}
s1.len(); // s1 失效了
}
more
Common Rust Lifetime Misconceptions
[Chinese]Rust生命周期常见误区
async
Unsafe
projects
windows:
windows dialog
Select a directory:
//extern crate wfd;
use wfd::{DialogParams, FOS_PICKFOLDERS};
fn main() {
let params = DialogParams {
options: FOS_PICKFOLDERS,
title: "Select a directory",
..Default::default()
};
let result = wfd::open_dialog(params);
println!("Selected Folder: {}", result.unwrap().selected_file_path.to_str().unwrap());
}
use wfd::{DialogParams, FOS_ALLOWMULTISELECT, DialogError};
fn main() {
let params = DialogParams {
file_types: vec![("DLL Files", "*.dll"), ("Executable Files", "*.exe;*.com;*.scr")],
default_extension: "dll",
default_folder: r"C:\Windows\System32",
file_name: "win32k.sys",
file_name_label: "Select some files!",
file_type_index: 1,
ok_button_label: "Custom OK",
options: FOS_ALLOWMULTISELECT,
title: "Test open file dialog",
.. Default::default()
};
match wfd::open_dialog(params) {
Ok(r) => {
for file in r.selected_file_paths {
println!("{}", file.to_str().unwrap());
}
}
Err(e) => match e {
DialogError::UserCancelled => {
println!("User cancelled dialog");
}
DialogError::HResultFailed { hresult, error_method } => {
println!("HResult Failed - HRESULT: {:X}, Method: {}", hresult, error_method);
}
}
}
}
walkdir:
extern crate walkdir;
use walkdir::WalkDir;
fn main() {
for e in WalkDir::new(".").into_iter().filter_map(|e| e.ok()) {
if e.metadata().unwrap().is_file() {
println!("{}", e.path().display());
}
}
}
compare file old or new
fn is_input_file_outdated<P1, P2>(input: P1, output: P2) -> io::Result<bool>
where
P1: AsRef<Path>,
P2: AsRef<Path>,
{
let out_meta = fs::metadata(output);
if let Ok(meta) = out_meta {
let output_mtime = meta.modified()?;
// if input file is more recent than our output, we are outdated
let input_meta = fs::metadata(input)?;
let input_mtime = input_meta.modified()?;
Ok(input_mtime > output_mtime)
} else {
// output file not found, we are outdated
Ok(true)
}
}
A number tool.
Explanation
A lazy of men will be try to save themselves typing as did not found his calculator.
Buying and selling of stocks, it has Initial minimum. The number is ten thousand.
fn main() {
use std::env;
let mut k = env::args();
if k.len() == 1 {
println!("Must be input a price!");
return;
}
let mut s = "".to_string();
match k.nth(1) {
Some(x) => s = x,
_ => return,
}
let num = strtof32(s.clone());
if num == 0.0 {
println!("Error, {}", s);
return;
}
println!("Input is {}, and get {}", s, (10000.0 / num) as u32);
}
pub fn strtof32<T: AsRef<str>>(s: T) -> f32 {
let picks: String =s.as_ref().chars().filter(|x| x.is_digit(10)||*x=='.').collect();
if picks.len() == 0 {
return 0.0;
}
picks.parse::<f32>().unwrap_or(0.0)
}
patterns
GoF 的 23 种设计模式的分类,现在对各个模式的功能进行介绍。
- 单例(Singleton)模式:某个类只能生成一个实例,该类提供了一个全局访问点供外部获取该实例,其拓展是有限多例模式。
- 原型(Prototype)模式:将一个对象作为原型,通过对其进行复制而克隆出多个和原型类似的新实例。
- 工厂方法(Factory Method)模式:定义一个用于创建产品的接口,由子类决定生产什么产品。
- 抽象工厂(AbstractFactory)模式:提供一个创建产品族的接口,其每个子类可以生产一系列相关的产品。
- 建造者(Builder)模式:将一个复杂对象分解成多个相对简单的部分,然后根据不同需要分别创建它们,最后构建成该复杂对象。
- 代理(Proxy)模式:为某对象提供一种代理以控制对该对象的访问。即客户端通过代理间接地访问该对象,从而限制、增强或修改该对象的一些特性。
- 适配器(Adapter)模式:将一个类的接口转换成客户希望的另外一个接口,使得原本由于接口不兼容而不能一起工作的那些类能一起工作。
- 桥接(Bridge)模式:将抽象与实现分离,使它们可以独立变化。它是用组合关系代替继承关系来实现,从而降低了抽象和实现这两个可变维度的耦合度。
- 装饰(Decorator)模式:动态的给对象增加一些职责,即增加其额外的功能。
- 外观(Facade)模式:为多个复杂的子系统提供一个一致的接口,使这些子系统更加容易被访问。
- 享元(Flyweight)模式:运用共享技术来有效地支持大量细粒度对象的复用。
- 组合(Composite)模式:将对象组合成树状层次结构,使用户对单个对象和组合对象具有一致的访问性。
- 模板方法(TemplateMethod)模式:定义一个操作中的算法骨架,而将算法的一些步骤延迟到子类中,使得子类可以不改变该算法结构的情况下重定义该算法的某些特定步骤。
- 策略(Strategy)模式:定义了一系列算法,并将每个算法封装起来,使它们可以相互替换,且算法的改变不会影响使用算法的客户。
- 命令(Command)模式:将一个请求封装为一个对象,使发出请求的责任和执行请求的责任分割开。
- 职责链(Chain of Responsibility)模式:把请求从链中的一个对象传到下一个对象,直到请求被响应为止。通过这种方式去除对象之间的耦合。
- 状态(State)模式:允许一个对象在其内部状态发生改变时改变其行为能力。
- 观察者(Observer)模式:多个对象间存在一对多关系,当一个对象发生改变时,把这种改变通知给其他多个对象,从而影响其他对象的行为。
- 中介者(Mediator)模式:定义一个中介对象来简化原有对象之间的交互关系,降低系统中对象间的耦合度,使原有对象之间不必相互了解。
- 迭代器(Iterator)模式:提供一种方法来顺序访问聚合对象中的一系列数据,而不暴露聚合对象的内部表示。
- 访问者(Visitor)模式:在不改变集合元素的前提下,为一个集合中的每个元素提供多种访问方式,即每个元素有多个访问者对象访问。
- 备忘录(Memento)模式:在不破坏封装性的前提下,获取并保存一个对象的内部状态,以便以后恢复它。
- 解释器(Interpreter)模式:提供如何定义语言的文法,以及对语言句子的解释方法,即解释器。
建造者模式(Builder)
概述
构建者模式是一种设计模式,提供一种灵活的解决方案,已解决面向对象程序设计中的各种对象创建问题。Builder设计模式的目的是将复杂对象的构造与其表示分离开来。是"是四人帮"设计模式之一[wiki]。建造者模式是一种创建型设计模式,使你能够分步骤创建复杂对象。该模式允许你使用相同的创建代码生成不同类型和形式的对象。
定义:Builder设计模式的目的是将复杂对象的构造与其表示分离开来。通过这样做,同样的构造过程可以创建不同的表示。
历史
假如有一个复杂的对象,需要对其进行构造时需要对诸多成员变量和嵌套对象进行繁杂的初始化工作。有时这些初始化代码通常深藏于一个包含众多参数且让人看不懂的构造函数中;或者这些代码散落在客户端代码的多个位置。
- 例如,创建一个房子,不同种类的房子有不同的风格,为每一种类型的房子创建一个子类,这可能会导致程序变得过于复杂。
- 或者无需生成子类,但是需要创建一个包括所有可能参数的超级构造函数,并用它来控制房屋对象的创建。这样虽然可以避免生成子类,但是会造成当拥有大量输入参数的构造函数不是每次都要全部用上。通常情况下,绝大部分的参数都没有使用,这使得对于构造函数的调用十分不简洁。
建造者模式 的使用
建造者模式建议将对象构造的代码从产品类中抽取出来,并将其放在一个名为生成器的独立对象中。每次创建对象时,都需要通过生成器对象执行一系列步骤。重点在于无需调用所有步骤,而只需调用创建特定对象配置所需的那些步骤。
适用场景
- 使用建造者设计模式可以避免“重叠构造函数”的出现。
- 假设复杂函数中有十几个可选参数,那么调用这些函数会非常不方便,因此需要重载这个构造函数,新建几个只有较少参数的简化版本。
- 建造者设计模式让你可以分步骤生成对象,而且允许你仅适用必须的步骤。
- 当使用代码创建不同形式的产品时,可使用生成器模式
- 如果你需要创建各种形式的产品,他们的制造过程相似且仅有细节上的差异,此时可使用生成器模式。
- 基本生成器接口中定义了所有可能的制造步骤,具体生成器将实现这些步骤来制造特定形式的产品。
- 使用构造者模式构造其他复杂对象
- 构造者模式让你能分步骤构造产品,你可以延迟执行某些步骤而不会影响最终产品。
优点
- 可以分步骤创建对象,暂缓创建步骤或者递归运行创建步骤。
- 生成不同形式的产品,你可以复用相同的制造代码
- 单一职责原则,可以将复杂构造代码从产品的业务逻辑中分离出来。
缺点
由于该模式需要新增多个类,因此代码整体复杂程度会有所增加。
描述
通过使用构建者助手创建一个对象。
例子
fn main() { let foo = Foo { bar: String::from("Y"), }; let foo_from_builder = FooBuilder::new().name(String::from("Y")).build(); println!("foo = {:?}", foo); println!("foo from builfer = {:?}", foo_from_builder); } #[derive(Debug, PartialEq)] pub struct Foo { // lots of complicated fields bar : String, } pub struct FooBuilder { // Probably lots of optional fields. bar: String, } impl FooBuilder { pub fn new() -> Self { // set the minimally required fields of Foo. Self { bar: String::from("x"), } } pub fn name(mut self, bar: String) -> FooBuilder { // set the name on the builder iteself, // and return the builder by value. self.bar = bar; self } // if we can get away with not consuming the builder here, that is an // advantage. It means we can use the FooBuilder as a template for constructing many Foo. pub fn build(self) -> Foo { // Create a Foo from Foo the FooBuilder, applying all settings in FooBuilder to Foo. Foo { bar: self.bar } } }
// Rust 编程之道. P234 struct Circle { x: f64, y: f64, radius: f64, } struct CircleBuilder { x: f64, y: f64, radius: f64, } impl Circle { fn area(&self) -> f64 { std::f64::consts::PI * (self.radius * self.radius) } fn new() -> CircleBuilder { CircleBuilder { x: 0.0, y: 0.0, radius: 1.0, } } } impl CircleBuilder { fn x(&mut self, coordinate: f64) -> &mut CircleBuilder { self.x = coordinate; self } fn y(&mut self, coordinate: f64) -> &mut CircleBuilder { self.y = coordinate; self } fn radius(&mut self, radius: f64) -> &mut CircleBuilder { self.radius = radius; self } fn build(&self) -> Circle { Circle { x: self.x, y: self.y, radius: self.radius, } } } fn main() { let c = Circle::new().x(1.0).y(2.0).radius(2.0).build(); println!("area = {:?}", c.area()); println!("c.x = {:?}", c.x); println!("c.y = {:?}", c.y); }
动机
当你需要许多不同的构造函数或者当构造有副作用时,这种方法有用。
优点
将构造方法与其他方法分离。
防止构造函数的扩散
可用于单次初始化以及更加复杂的构造。
缺点
比直接创建结构对象或简单的的构造函数更复杂。
讨论
这种模式在Rust(以及简单对象)中比在其他许多语言中更常见,这是因为Rust缺乏重载。由于你只能使用给定名称的单个方法,因此在Rust中使用多个构造函数要比C++、Java或其他语言好。
这种模式通常用于构建器对象本身就很有用的地方,而不仅仅是一个构建器。例如:std::process::Command 是Child的构建器。在这种情况下,不使用T和TBuilder的命名模式。
该示例通过值获取并返回生成器。接受并返回构建器作为可变引用通常更符合人体工程学(并且更有效)。
#![allow(unused)] fn main() { let mut fb = FooBuilder::new(); fb.a(); fb.b(); let f = fb.builder(); }
以及FooBuilder::new().a().b().builder()样式。
参见
- Description in the style guide
- derive_builder, a crate for automatically implementing this pattern while avoiding the boilerplate.
- Constructor pattern for when construction is simpler.
- Builder pattern (wikipedia)
- Construction of complex values
- Rust编程之道 ch7,p234
项目中的使用
Tokio 中的建造者模式 Struct tokio::runtime::Builder
#![allow(unused)] fn main() { pub struct Builder { /// Runtime type kind: Kind, /// Whether or not to enable the I/O driver enable_io: bool, /// Whether or not to enable the time driver enable_time: bool, /// The number of worker threads, used by Runtime. /// /// Only used when not using the current-thread executor. worker_threads: Option<usize>, /// Cap on thread usage. max_blocking_threads: usize, /// Name fn used for threads spawned by the runtime. pub(super) thread_name: ThreadNameFn, /// Stack size used for threads spawned by the runtime. pub(super) thread_stack_size: Option<usize>, /// Callback to run after each thread starts. pub(super) after_start: Option<Callback>, /// To run before each worker thread stops pub(super) before_stop: Option<Callback>, /// Customizable keep alive timeout for BlockingPool pub(super) keep_alive: Option<Duration>, } pub fn new_current_thread() -> Builder // 设置current thread 类型 //Returns a new builder with the current thread scheduler selected. //Configuration methods can be chained on the return value. pub fn new_multi_thread() -> Builder // 设置 multi thread 类型 //This is supported on crate feature rt-multi-thread only. //Returns a new builder with the multi thread scheduler selected. //Configuration methods can be chained on the return value. pub fn enable_all(&mut self) -> &mut Self // Enables both I/O and time drivers. // Doing this is a shorthand for calling enable_io and enable_time individually. If additional components are added to Tokio in the future, enable_all will include these future components. pub fn worker_threads(&mut self, val: usize) -> &mut Self // 设置的runtime 用于工作的线程数 // Sets the number of worker threads the Runtime will use. // This should be a number between 0 and 32,768 though it is advised to keep this value on the smaller side. pub fn max_blocking_threads(&mut self, val: usize) -> &mut Self // 设置生成的用于阻塞操作的线程最大数 //Specifies limit for threads spawned by the Runtime used for blocking operations. //Similarly to the worker_threads, this number should be between 1 and 32,768. //The default value is 512. //Otherwise as worker_threads are always active, it limits additional threads (e.g. for blocking annotations). pub fn thread_name(&mut self, val: impl Into<String>) -> &mut Self // 设置线程的名字 //Sets name of threads spawned by the Runtime's thread pool. //The default name is "tokio-runtime-worker". // ..... pub fn build(&mut self) -> Result<Runtime> // 构造出tokio中的runtime结构 //Creates the configured Runtime. //The returned Runtime instance is ready to spawn tasks. //etc.. //example // build runtime let runtime = Builder::new_multi_thread() .worker_threads(4) .thread_name("my-custom-name") .thread_stack_size(3 * 1024 * 1024) .build() .unwrap(); }
从Builder的build函数可以知道Builder结构是Runtime的辅助结构体用来帮助构造Runtime的。
Futures 中的建造者设计模式 Struct futures::executor::ThreadPoolBuilder
#![allow(unused)] fn main() { /// A general-purpose thread pool for scheduling tasks that poll futures to /// completion. /// /// The thread pool multiplexes any number of tasks onto a fixed number of /// worker threads. /// /// This type is a clonable handle to the threadpool itself. /// Cloning it will only create a new reference, not a new threadpool. /// /// This type is only available when the `thread-pool` feature of this /// library is activated. #[cfg_attr(docsrs, doc(cfg(feature = "thread-pool")))] pub struct ThreadPool { state: Arc<PoolState>, } /// Thread pool configuration object. /// /// This type is only available when the `thread-pool` feature of this /// library is activated. #[cfg_attr(docsrs, doc(cfg(feature = "thread-pool")))] pub struct ThreadPoolBuilder { pool_size: usize, stack_size: usize, name_prefix: Option<String>, after_start: Option<Arc<dyn Fn(usize) + Send + Sync>>, before_stop: Option<Arc<dyn Fn(usize) + Send + Sync>>, } struct PoolState { tx: Mutex<mpsc::Sender<Message>>, rx: Mutex<mpsc::Receiver<Message>>, cnt: AtomicUsize, size: usize, } enum Message { Run(Task), Close, } impl ThreadPoolBuilder { /// Create a default thread pool configuration. /// /// See the other methods on this type for details on the defaults. pub fn new() -> Self { Self { pool_size: cmp::max(1, num_cpus::get()), stack_size: 0, name_prefix: None, after_start: None, before_stop: None, } } /// Set size of a future ThreadPool /// /// The size of a thread pool is the number of worker threads spawned. By /// default, this is equal to the number of CPU cores. /// /// # Panics /// /// Panics if `pool_size == 0`. pub fn pool_size(&mut self, size: usize) -> &mut Self { assert!(size > 0); self.pool_size = size; self } /// Set stack size of threads in the pool, in bytes. /// /// By default, worker threads use Rust's standard stack size. pub fn stack_size(&mut self, stack_size: usize) -> &mut Self { self.stack_size = stack_size; self } /// Set thread name prefix of a future ThreadPool. /// /// Thread name prefix is used for generating thread names. For example, if prefix is /// `my-pool-`, then threads in the pool will get names like `my-pool-1` etc. /// /// By default, worker threads are assigned Rust's standard thread name. pub fn name_prefix<S: Into<String>>(&mut self, name_prefix: S) -> &mut Self { self.name_prefix = Some(name_prefix.into()); self } /// Execute the closure `f` immediately after each worker thread is started, /// but before running any tasks on it. /// /// This hook is intended for bookkeeping and monitoring. /// The closure `f` will be dropped after the `builder` is dropped /// and all worker threads in the pool have executed it. /// /// The closure provided will receive an index corresponding to the worker /// thread it's running on. pub fn after_start<F>(&mut self, f: F) -> &mut Self where F: Fn(usize) + Send + Sync + 'static { self.after_start = Some(Arc::new(f)); self } /// Execute closure `f` just prior to shutting down each worker thread. /// /// This hook is intended for bookkeeping and monitoring. /// The closure `f` will be dropped after the `builder` is droppped /// and all threads in the pool have executed it. /// /// The closure provided will receive an index corresponding to the worker /// thread it's running on. pub fn before_stop<F>(&mut self, f: F) -> &mut Self where F: Fn(usize) + Send + Sync + 'static { self.before_stop = Some(Arc::new(f)); self } // 从ThreadBuilder的create函数可以看到ThreadPoolBuilder根据配置采纳数创建ThreadPool, 是ThreadPool的辅助结构体 /// Create a [`ThreadPool`](ThreadPool) with the given configuration. pub fn create(&mut self) -> Result<ThreadPool, io::Error> { let (tx, rx) = mpsc::channel(); let pool = ThreadPool { state: Arc::new(PoolState { tx: Mutex::new(tx), rx: Mutex::new(rx), cnt: AtomicUsize::new(1), size: self.pool_size, }), }; for counter in 0..self.pool_size { let state = pool.state.clone(); let after_start = self.after_start.clone(); let before_stop = self.before_stop.clone(); let mut thread_builder = thread::Builder::new(); if let Some(ref name_prefix) = self.name_prefix { thread_builder = thread_builder.name(format!("{}{}", name_prefix, counter)); } if self.stack_size > 0 { thread_builder = thread_builder.stack_size(self.stack_size); } thread_builder.spawn(move || state.work(counter, after_start, before_stop))?; } Ok(pool) } } }
从ThreadBuilder的create函数可以看到ThreadPoolBuilder根据配置采纳数创建ThreadPool, 是ThreadPool的辅助结构体
Surf中的建造者设计模式
/// Request Builder /// /// Provides an ergonomic way to chain the creation of a request. /// This is generally accessed as the return value from `surf::{method}()`, /// however [`Request::builder`](crate::Request::builder) is also provided. /// /// # Examples /// /// ```rust /// use surf::http::{Method, mime::HTML, Url}; /// # #[async_std::main] /// # async fn main() -> surf::Result<()> { /// let mut request = surf::post("https://httpbin.org/post") /// .body("<html>hi</html>") /// .header("custom-header", "value") /// .content_type(HTML) /// .build(); /// /// assert_eq!(request.take_body().into_string().await.unwrap(), "<html>hi</html>"); /// assert_eq!(request.method(), Method::Post); /// assert_eq!(request.url(), &Url::parse("https://httpbin.org/post")?); /// assert_eq!(request["custom-header"], "value"); /// assert_eq!(request["content-type"], "text/html;charset=utf-8"); /// # Ok(()) /// # } /// ``` /// /// ```rust /// use surf::http::{Method, Url}; /// # #[async_std::main] /// # async fn main() -> surf::Result<()> { /// let url = Url::parse("https://httpbin.org/post")?; /// let request = surf::Request::builder(Method::Post, url).build(); /// # Ok(()) /// # } /// ``` pub struct RequestBuilder { /// Holds the state of the request. req: Option<Request>, /// Hold an optional Client. client: Option<Client>, /// Holds the state of the `impl Future`. fut: Option<BoxFuture<'static, Result<Response>>>, } impl RequestBuilder { /// Create a new instance. /// /// This method is particularly useful when input URLs might be passed by third parties, and /// you don't want to panic if they're malformed. If URLs are statically encoded, it might be /// easier to use one of the shorthand methods instead. /// /// # Examples /// /// ```no_run /// # #[async_std::main] /// # async fn main() -> surf::Result<()> { /// use surf::http::{Method, Url}; /// /// let url = Url::parse("https://httpbin.org/get")?; /// let req = surf::RequestBuilder::new(Method::Get, url).build(); /// # Ok(()) } /// ``` pub fn new(method: Method, url: Url) -> Self { Self { req: Some(Request::new(method, url)), client: None, fut: None, } } pub(crate) fn with_client(mut self, client: Client) -> Self { self.client = Some(client); self } /// Sets a header on the request. /// /// # Examples /// /// ``` /// let req = surf::get("https://httpbin.org/get").header("header-name", "header-value").build(); /// assert_eq!(req["header-name"], "header-value"); /// ``` pub fn header(mut self, key: impl Into<HeaderName>, value: impl ToHeaderValues) -> Self { self.req.as_mut().unwrap().insert_header(key, value); self } /// Sets the Content-Type header on the request. /// /// # Examples /// /// ``` /// # use surf::http::mime; /// let req = surf::post("https://httpbin.org/post").content_type(mime::HTML).build(); /// assert_eq!(req["content-type"], "text/html;charset=utf-8"); /// ``` pub fn content_type(mut self, content_type: impl Into<Mime>) -> Self { self.req .as_mut() .unwrap() .set_content_type(content_type.into()); self } /// Sets the body of the request. /// /// # Examples /// /// ``` /// # #[async_std::main] /// # async fn main() -> surf::Result<()> { /// use serde_json::json; /// let mut req = surf::post("https://httpbin.org/post").body(json!({ "any": "Into<Body>"})).build(); /// assert_eq!(req.take_body().into_string().await.unwrap(), "{\"any\":\"Into<Body>\"}"); /// # Ok(()) /// # } /// ``` pub fn body(mut self, body: impl Into<Body>) -> Self { self.req.as_mut().unwrap().set_body(body); self } /// Set the URL querystring. /// /// # Examples /// /// ```no_run /// # use serde::{Deserialize, Serialize}; /// # #[async_std::main] /// # async fn main() -> surf::Result<()> { /// #[derive(Serialize, Deserialize)] /// struct Index { /// page: u32 /// } /// /// let query = Index { page: 2 }; /// let mut req = surf::get("https://httpbin.org/get").query(&query)?.build(); /// assert_eq!(req.url().query(), Some("page=2")); /// assert_eq!(req.url().as_str(), "https://httpbin.org/get?page=2"); /// # Ok(()) } /// ``` pub fn query(mut self, query: &impl Serialize) -> std::result::Result<Self, Error> { self.req.as_mut().unwrap().set_query(query)?; Ok(self) } /// Submit the request and get the response body as bytes. /// /// # Examples /// /// ```no_run /// # #[async_std::main] /// # async fn main() -> surf::Result<()> { /// let bytes = surf::get("https://httpbin.org/get").recv_bytes().await?; /// assert!(bytes.len() > 0); /// # Ok(()) } /// ``` pub async fn recv_bytes(self) -> Result<Vec<u8>> { let mut res = self.send().await?; Ok(res.body_bytes().await?) } /// Submit the request and get the response body as a string. /// /// # Examples /// /// ```no_run /// # #[async_std::main] /// # async fn main() -> surf::Result<()> { /// let string = surf::get("https://httpbin.org/get").recv_string().await?; /// assert!(string.len() > 0); /// # Ok(()) } /// ``` pub async fn recv_string(self) -> Result<String> { let mut res = self.send().await?; Ok(res.body_string().await?) } /// Submit the request and decode the response body from json into a struct. /// /// # Examples /// /// ```no_run /// # use serde::{Deserialize, Serialize}; /// # #[async_std::main] /// # async fn main() -> surf::Result<()> { /// #[derive(Deserialize, Serialize)] /// struct Ip { /// ip: String /// } /// /// let uri = "https://api.ipify.org?format=json"; /// let Ip { ip } = surf::get(uri).recv_json().await?; /// assert!(ip.len() > 10); /// # Ok(()) } /// ``` pub async fn recv_json<T: serde::de::DeserializeOwned>(self) -> Result<T> { let mut res = self.send().await?; Ok(res.body_json::<T>().await?) } /// Submit the request and decode the response body from form encoding into a struct. /// /// # Errors /// /// Any I/O error encountered while reading the body is immediately returned /// as an `Err`. /// /// If the body cannot be interpreted as valid json for the target type `T`, /// an `Err` is returned. /// /// # Examples /// /// ```no_run /// # use serde::{Deserialize, Serialize}; /// # #[async_std::main] /// # async fn main() -> surf::Result<()> { /// #[derive(Deserialize, Serialize)] /// struct Body { /// apples: u32 /// } /// /// let url = "https://api.example.com/v1/response"; /// let Body { apples } = surf::get(url).recv_form().await?; /// # Ok(()) } /// ``` pub async fn recv_form<T: serde::de::DeserializeOwned>(self) -> Result<T> { let mut res = self.send().await?; Ok(res.body_form::<T>().await?) } // 从build函数可以知道最后RequestBuilder是Request的辅助结构体,用来构造返回Request // 这个函数返回的是Request /// Return the constructed `Request`. pub fn build(self) -> Request { self.req.unwrap() } /// Create a `Client` and send the constructed `Request` from it. pub async fn send(mut self) -> Result<Response> { self.client .take() .unwrap_or_else(Client::new_shared_or_panic) .send(self.build()) .await } }
从build函数可以知道最后RequestBuilder是Request的辅助结构体,用来构造返回Request
Reqwest中的建造者设计模式
#![allow(unused)] fn main() { /// A request which can be executed with `Client::execute()`. pub struct Request { method: Method, url: Url, headers: HeaderMap, body: Option<Body>, timeout: Option<Duration>, } /// A builder to construct the properties of a `Request`. /// /// To construct a `RequestBuilder`, refer to the `Client` documentation. #[must_use = "RequestBuilder does nothing until you 'send' it"] pub struct RequestBuilder { client: Client, request: crate::Result<Request>, } impl Request { /// Constructs a new request. #[inline] pub fn new(method: Method, url: Url) -> Self { Request { method, url, headers: HeaderMap::new(), body: None, timeout: None } } /// Get the method. #[inline] pub fn method(&self) -> &Method { &self.method } /// Get a mutable reference to the method. #[inline] pub fn method_mut(&mut self) -> &mut Method { &mut self.method } /// Get the url. #[inline] pub fn url(&self) -> &Url { &self.url } /// Get a mutable reference to the url. #[inline] pub fn url_mut(&mut self) -> &mut Url { &mut self.url } /// Get the headers. #[inline] pub fn headers(&self) -> &HeaderMap { &self.headers } /// Get a mutable reference to the headers. #[inline] pub fn headers_mut(&mut self) -> &mut HeaderMap { &mut self.headers } /// Get the body. #[inline] pub fn body(&self) -> Option<&Body> { self.body.as_ref() } /// Get a mutable reference to the body. #[inline] pub fn body_mut(&mut self) -> &mut Option<Body> { &mut self.body } /// Get the timeout. #[inline] pub fn timeout(&self) -> Option<&Duration> { self.timeout.as_ref() } /// Get a mutable reference to the timeout. #[inline] pub fn timeout_mut(&mut self) -> &mut Option<Duration> { &mut self.timeout } /// Attempt to clone the request. /// /// `None` is returned if the request can not be cloned, i.e. if the body is a stream. pub fn try_clone(&self) -> Option<Request> { let body = match self.body.as_ref() { Some(ref body) => Some(body.try_clone()?), None => None, }; let mut req = Request::new(self.method().clone(), self.url().clone()); *req.timeout_mut() = self.timeout().cloned(); *req.headers_mut() = self.headers().clone(); req.body = body; Some(req) } pub(super) fn pieces(self) -> (Method, Url, HeaderMap, Option<Body>, Option<Duration>) { (self.method, self.url, self.headers, self.body, self.timeout) } } impl RequestBuilder { pub(super) fn new(client: Client, request: crate::Result<Request>) -> RequestBuilder { let mut builder = RequestBuilder { client, request }; let auth = builder .request .as_mut() .ok() .and_then(|req| extract_authority(&mut req.url)); if let Some((username, password)) = auth { builder.basic_auth(username, password) } else { builder } } /// Add a `Header` to this Request. pub fn header<K, V>(self, key: K, value: V) -> RequestBuilder where HeaderName: TryFrom<K>, <HeaderName as TryFrom<K>>::Error: Into<http::Error>, HeaderValue: TryFrom<V>, <HeaderValue as TryFrom<V>>::Error: Into<http::Error>, { self.header_sensitive(key, value, false) } /// Add a `Header` to this Request with ability to define if header_value is sensitive. fn header_sensitive<K, V>(mut self, key: K, value: V, sensitive: bool) -> RequestBuilder where HeaderName: TryFrom<K>, <HeaderName as TryFrom<K>>::Error: Into<http::Error>, HeaderValue: TryFrom<V>, <HeaderValue as TryFrom<V>>::Error: Into<http::Error>, { let mut error = None; if let Ok(ref mut req) = self.request { match <HeaderName as TryFrom<K>>::try_from(key) { Ok(key) => match <HeaderValue as TryFrom<V>>::try_from(value) { Ok(mut value) => { value.set_sensitive(sensitive); req.headers_mut().append(key, value); } Err(e) => error = Some(crate::error::builder(e.into())), }, Err(e) => error = Some(crate::error::builder(e.into())), }; } if let Some(err) = error { self.request = Err(err); } self } /// Add a set of Headers to the existing ones on this Request. /// /// The headers will be merged in to any already set. pub fn headers(mut self, headers: crate::header::HeaderMap) -> RequestBuilder { if let Ok(ref mut req) = self.request { crate::util::replace_headers(req.headers_mut(), headers); } self } /// Enable HTTP basic authentication. pub fn basic_auth<U, P>(self, username: U, password: Option<P>) -> RequestBuilder where U: fmt::Display, P: fmt::Display, { let mut header_value = b"Basic ".to_vec(); { let mut encoder = Base64Encoder::new(&mut header_value, base64::STANDARD); // The unwraps here are fine because Vec::write* is infallible. write!(encoder, "{}:", username).unwrap(); if let Some(password) = password { write!(encoder, "{}", password).unwrap(); } } self.header_sensitive(crate::header::AUTHORIZATION, header_value, true) } /// Enable HTTP bearer authentication. pub fn bearer_auth<T>(self, token: T) -> RequestBuilder where T: fmt::Display, { let header_value = format!("Bearer {}", token); self.header_sensitive(crate::header::AUTHORIZATION, header_value, true) } /// Set the request body. pub fn body<T: Into<Body>>(mut self, body: T) -> RequestBuilder { if let Ok(ref mut req) = self.request { *req.body_mut() = Some(body.into()); } self } /// Enables a request timeout. /// /// The timeout is applied from when the request starts connecting until the /// response body has finished. It affects only this request and overrides /// the timeout configured using `ClientBuilder::timeout()`. pub fn timeout(mut self, timeout: Duration) -> RequestBuilder { if let Ok(ref mut req) = self.request { *req.timeout_mut() = Some(timeout); } self } /// Sends a multipart/form-data body. /// /// ``` /// # use reqwest::Error; /// /// # async fn run() -> Result<(), Error> { /// let client = reqwest::Client::new(); /// let form = reqwest::multipart::Form::new() /// .text("key3", "value3") /// .text("key4", "value4"); /// /// /// let response = client.post("your url") /// .multipart(form) /// .send() /// .await?; /// # Ok(()) /// # } /// ``` #[cfg(feature = "multipart")] pub fn multipart(self, mut multipart: multipart::Form) -> RequestBuilder { let mut builder = self.header( CONTENT_TYPE, format!("multipart/form-data; boundary={}", multipart.boundary()).as_str(), ); builder = match multipart.compute_length() { Some(length) => builder.header(CONTENT_LENGTH, length), None => builder, }; if let Ok(ref mut req) = builder.request { *req.body_mut() = Some(multipart.stream()) } builder } /// Modify the query string of the URL. /// /// Modifies the URL of this request, adding the parameters provided. /// This method appends and does not overwrite. This means that it can /// be called multiple times and that existing query parameters are not /// overwritten if the same key is used. The key will simply show up /// twice in the query string. /// Calling `.query([("foo", "a"), ("foo", "b")])` gives `"foo=a&foo=b"`. /// /// # Note /// This method does not support serializing a single key-value /// pair. Instead of using `.query(("key", "val"))`, use a sequence, such /// as `.query(&[("key", "val")])`. It's also possible to serialize structs /// and maps into a key-value pair. /// /// # Errors /// This method will fail if the object you provide cannot be serialized /// into a query string. pub fn query<T: Serialize + ?Sized>(mut self, query: &T) -> RequestBuilder { let mut error = None; if let Ok(ref mut req) = self.request { let url = req.url_mut(); let mut pairs = url.query_pairs_mut(); let serializer = serde_urlencoded::Serializer::new(&mut pairs); if let Err(err) = query.serialize(serializer) { error = Some(crate::error::builder(err)); } } if let Ok(ref mut req) = self.request { if let Some("") = req.url().query() { req.url_mut().set_query(None); } } if let Some(err) = error { self.request = Err(err); } self } /// Send a form body. pub fn form<T: Serialize + ?Sized>(mut self, form: &T) -> RequestBuilder { let mut error = None; if let Ok(ref mut req) = self.request { match serde_urlencoded::to_string(form) { Ok(body) => { req.headers_mut().insert( CONTENT_TYPE, HeaderValue::from_static("application/x-www-form-urlencoded"), ); *req.body_mut() = Some(body.into()); } Err(err) => error = Some(crate::error::builder(err)), } } if let Some(err) = error { self.request = Err(err); } self } /// Send a JSON body. /// /// # Optional /// /// This requires the optional `json` feature enabled. /// /// # Errors /// /// Serialization can fail if `T`'s implementation of `Serialize` decides to /// fail, or if `T` contains a map with non-string keys. #[cfg(feature = "json")] pub fn json<T: Serialize + ?Sized>(mut self, json: &T) -> RequestBuilder { let mut error = None; if let Ok(ref mut req) = self.request { match serde_json::to_vec(json) { Ok(body) => { req.headers_mut() .insert(CONTENT_TYPE, HeaderValue::from_static("application/json")); *req.body_mut() = Some(body.into()); } Err(err) => error = Some(crate::error::builder(err)), } } if let Some(err) = error { self.request = Err(err); } self } /// Disable CORS on fetching the request. /// /// # WASM /// /// This option is only effective with WebAssembly target. /// /// The [request mode][mdn] will be set to 'no-cors'. /// /// [mdn]: https://developer.mozilla.org/en-US/docs/Web/API/Request/mode pub fn fetch_mode_no_cors(self) -> RequestBuilder { self } // 从RequestBuilder的build函数可以知道,RequestBuilder是用来帮助构造Request的辅助结构体 /// Build a `Request`, which can be inspected, modified and executed with /// `Client::execute()`. pub fn build(self) -> crate::Result<Request> { self.request } /// Constructs the Request and sends it to the target URL, returning a /// future Response. /// /// # Errors /// /// This method fails if there was an error while sending request, /// redirect loop was detected or redirect limit was exhausted. /// /// # Example /// /// ```no_run /// # use reqwest::Error; /// # /// # async fn run() -> Result<(), Error> { /// let response = reqwest::Client::new() /// .get("https://hyper.rs") /// .send() /// .await?; /// # Ok(()) /// # } /// ``` pub fn send(self) -> impl Future<Output = Result<Response, crate::Error>> { match self.request { Ok(req) => self.client.execute_request(req), Err(err) => Pending::new_err(err), } } /// Attempt to clone the RequestBuilder. /// /// `None` is returned if the RequestBuilder can not be cloned, /// i.e. if the request body is a stream. /// /// # Examples /// /// ``` /// # use reqwest::Error; /// # /// # fn run() -> Result<(), Error> { /// let client = reqwest::Client::new(); /// let builder = client.post("http://httpbin.org/post") /// .body("from a &str!"); /// let clone = builder.try_clone(); /// assert!(clone.is_some()); /// # Ok(()) /// # } /// ``` pub fn try_clone(&self) -> Option<RequestBuilder> { self.request .as_ref() .ok() .and_then(|req| req.try_clone()) .map(|req| RequestBuilder { client: self.client.clone(), request: Ok(req), }) } } }
从RequestBuilder的build函数可以知道,RequestBuilder是用来帮助构造Request的辅助结构体。
参考链接:
https://docs.rs/tokio/1.1.0/tokio/runtime/struct.Builder.html
https://docs.rs/reqwest/0.11.0/src/reqwest/async_impl/request.rs.html#36-39
https://github.com/http-rs/surf/blob/31315743b91ff003231183c1ec5a3cd2b698c58a/src/request_builder.rs
https://docs.rs/futures/0.3.12/futures/executor/struct.ThreadPoolBuilder.html
document
rustup doc
--alloc The Rust core allocation and collections library
--book The Rust Programming Language book
--cargo The Cargo Book
--core The Rust Core Library
--edition-guide The Rust Edition Guide
--embedded-book The Embedded Rust Book
-h, --help Prints help information
--nomicon The Dark Arts of Advanced and Unsafe Rust Programming
--path Only print the path to the documentation
--proc_macro A support library for macro authors when defining new macros
--reference The Rust Reference
--rust-by-example A collection of runnable examples that illustrate various Rust concepts and standard
libraries
--rustc The compiler for the Rust programming language
--rustdoc Generate documentation for Rust projects
--std Standard library API documentation
--test Support code for rustc's built in unit-test and micro-benchmarking framework
--unstable-book The Unstable Book
metaprogramming
proc_macro
await
own
cli
gui
iced
bevy
metadata
# why what how
essential
what where why
name start class
supplementary
rustup doc
rustup doc --help
--alloc The Rust core allocation and collections library
--book The Rust Programming Language book
--cargo The Cargo Book
--core The Rust Core Library
--edition-guide The Rust Edition Guide
--embedded-book The Embedded Rust Book
-h, --help Prints help information
--nomicon The Dark Arts of Advanced and Unsafe Rust Programming
--path Only print the path to the documentation
--proc_macro A support library for macro authors when defining new macros
--reference The Rust Reference
--rust-by-example A collection of runnable examples that illustrate various Rust concepts and standard
libraries
--rustc The compiler for the Rust programming language
--rustdoc Generate documentation for Rust projects
--std Standard library API documentation
--test Support code for rustc's built in unit-test and micro-benchmarking framework
--unstable-book The Unstable Book
Record
rusqlite
unsuccessfully
gtk4-rs
error: failed to run custom build command for `pango-sys v0.13.0
error: failed to get glib-sys
as a dependency of package `gtk-rs-examples4 v0.0.1