自从做了 https://github.com/chainx-org/ChainX
项目以后,主力语言就转到了 Rust,今天刚好这个文章,比较剪短,跟大家分享一下。
在开始之前,跟大家简单介绍 ChainX 项目。ChainX 是一个基于 substrate 专注于区块链资产跨链项目,目前已经实现了 BTC 跨链, 可以在我们的测试网进行充值体验,如何参与测试网请点击这里。
我们将会在最近上线主网,并进行开源,欢迎有识之士进行关注,项目地址是: https://github.com/chainx-org/ChainX
。此外,也欢迎开发者加入我们的开发群,有兴趣的可以私信我加群。
ChainX 也在不断招人,运营,技术都招,如果有任何想法,欢迎与我联系,期待大牛加入!
好了,开始今天的“正题”:
更“护眼”的 print 调试
当我们用 print 大法进行调试的时候,经常会用到 :?
格式化操作符。但是除此以外,还有另外一些非常好用的操作符!另一个非常有用的就是 :#?
,它会自动加入换行和缩进来增强输出的可读性。
#[derive(Debug)]
struct Foo {
x: i32,
y: i32,
}
let foo = Foo { x: 1, y: 2 };
println!("Simple debug:\n{:?}", foo);
println!("Pretty debug:\n{:#?}", foo);
Simple debug:
Foo { x: 1, y: 2 }
Pretty debug:
Foo {
x: 1,
y: 2,
}
关于调试,还可以了解一下最近新加的 dbg!
宏 https://doc.rust-lang.org/std/macro.dbg.html 。
unimplemented!
有时候,你可能会想要一个不用进行完整实现的函数。比如,你可能想要一些方法的测试,又或者你想要为以后的开发保留某个 feature,这时 unimplemented!
就会派上用场。如果想要的类型是什么,unimplemented!
都会被展开为能够编译的表达式。
enum VerySimpleList<T> {
Empty,
Elem(T, Box<VerySimpleList>),
}
impl<T> VerySimpleList<T> {
fn len(&self) -> usize {
match self {
VerySimpleList::Empty => 0,
VerySimpleList::Elem(..) => unimplemented!(),
}
}
}
..
结构体字面操作符
有时候,你想要部分地复制一个结构体,也就是里面有部分字段不一样,但是其他字段保留复制结构体里面的内容。尽管你可以通过手动 clone 然后进行修改,但是还有更简单的方式!通过 ..
操作符后面跟着这个结构体的另一个实例,剩下的字段就会用后面这个实例的字段填充。此外,它并不要求结构体实现 Clone
约束。
#[derive(Debug, Default)]
struct Foo {
x: i32,
y: i32,
}
let a = Foo { x: 1, y: 2 };
let b = Foo { x: 2, ..a };
let c = Foo { x: 2, ..Default::default() };
模式匹配 guard
有时当使用模式匹配时,你所匹配的模式并没有跟你想要处理的格式完美匹配。比如,你可能会写这样的代码:
fn divide_opt(x: Option<i32>, y: Option<i32>) -> Option<i32> {
match (x, y) {
(Some(i), Some(0)) => None
(Some(i), Some(j)) => Some(i / j)
_ => None,
}
}
你大可以把这两种情况组合起来:
fn divide_opt(x: Option<i32>, y: Option<i32>) -> Option<i32> {
match (x, y) {
(Some(i), Some(j)) => {
if j == 0 {
None
} else {
Some(i / j)
}
}
_ => None,
}
}
但是,还有更好的方式!模式后面可以跟着一个条件表达式,叫做 “guard":
fn divide_opt(x: Option<i32>, y: Option<i32>) -> Option<i32> {
match (x, y) {
(Some(i), Some(j)) if j != 0 => Some(i / j),
_ => None,
}
}
填充格式操作符
想要在 Rust 里面进行左侧填充字符?不需要任何额外的包就可以实现!只要用 :>
操作符后面跟上一个长度,然后就好了!
let score1 = 100;
let score2 = 1000;
let score3 = 10000;
println!("{:>5}", score1);
println!("{:>5}", score2);
println!("{:>5}", score3);
这会打印出:
first player: padded!
second player: padded!
third player: padded!
等一下,还没完!如果你想要两边填充,也有办法, :^
后面跟上填充后的字符串宽度就行了:
let padded = "padded";
println!("[{:^10}]", padded)
这会打印出:
[ padded ]
你知道还可以用不同的字符进行填充吗?只要在箭头前面指定填充字符就行了!
let title = "SCORES";
let player1 = "first player:";
let player2 = "second player:";
let player3 = "third player:";
let score1 = 100;
let score2 = 1000;
let score3 = 10000;
println!("{:_^20}", title);
println!("{:<14} {:>5}", player1, score1);
println!("{:<14} {:>5}", player2, score2);
println!("{:<14} {:>5}", player3, score3);
这会打印出:
_______SCORES_______
first player: 100
second player: 1000
third player: 10000