rust 笔记 8:rust & tokio 杂记

1. 在 tokio::main 宏中指定 runtime 的线程个数

默认是多线程 runtime,worker 个数为 CPU 的核心数。可以通过在 tokio::main 中增加如下内容来制定 worker 个数:

#[tokio::main(flavor = "multi_thread", worker_threads = 10)]

或者切换成单线程 runtime:

#[tokio::main(flavor = "current_thread")]

其实鼠标 hover 到 tokio::main 上然后读注释就行了,这个在 tokio 的注释里写清楚了。

2. 在 release 版本里生成 debuginfo

在 Cargo.toml 里加上:

[profile.release]
debug = true
strip = false

第一行会在 release 版本的二进制里添加 debuginfo (文件名、行数、函数名等);第二行会关闭 release 编译时对符号的裁剪。
二者结合起来,可以让 release 版本的二进制程序更容易通过 gdb 观察调用栈等方式 debug,也能够通过 flamegraph 等生成火焰图。

2.1 题外话之 flamegraph 使用

https://github.com/brendangregg/FlameGraph

sudo perf record -e cpu-clock -g --call-graph dwarf -p 28591
perf script -i perf.data &> perf.unfold
./stackcollapse-perf.pl perf.unfold &> perf.folded
./flamegraph.pl perf.folded > perf.svg

注意如果抓出来的没有 debuginfo 可能是因为没有带 dwarf

参考:
https://www.cnblogs.com/happyliu/p/6142929.html
https://stackoverflow.com/questions/10933408/how-can-i-get-perf-to-find-symbols-in-my-program

3. AsyncRead & AsyncWrite 统计问题

在业务代码中对 TCP 做收发字节数的统计时,会发现一个比较诡异的问题:AsyncRead 没有返回读了多少字节的接口,但是 AsyncWrite 却是原生支持统计的。

pub trait AsyncWrite {
    // 返回的 Ok(usize) 就代表成功写入了多少字节到缓冲中
    fn poll_write(
        self: Pin<&mut Self>,
        cx: &mut Context<'_>,
        buf: &[u8],
    ) -> Poll<Result<usize, Error>>;
    // .....
}

pub trait AsyncRead {
    fn poll_read(
        self: Pin<&mut Self>,
        cx: &mut Context<'_>,
        buf: &mut ReadBuf<'_>,
    ) -> Poll<Result<()>>;
}

这个设计本身是很正常的,毕竟 poll_read 要求我们传入一个 ReadBufReadBuf 的所有方自然可以进行读取字节数的统计。但是,对于一些高度封装的库,这种统计接口并没有暴露出来,例如先用 tokio 创建 TcpStream,然后用 native-tls 进行 tls 握手的场景。

在这种情况下,就只能在 TcpStream 上面再封一层 Buffer 了,需要为这个新的 Buffer 实现 AsyncReadAsyncWrite。后者的实现很简单,直接调用底层 TcpStream 的接口即可;前者稍微复杂些,需要在读写前后分别调用 ReadBuffilled 方法,做减法来判断读了多少字节。

4. 对引用的比较与 std::ptr::eq

使用 == 运算符,实际上会调用对应类型的 PartialEq 实现。通常,我们会使用 #[derive(PartialEq)] 来为我们自定义的枚举或结构实现 PartialEq,而这样生成的代码逻辑会是逐个比较成员的值是否相等

当我们想要比较两个引用是否指向同一个实例时,可能会想要尝试使用 == 运算符。然而事与愿违,这种操作实际上也会变成使用 PartialEq 来比较其值是否相等。为什么呢?看看 PartialEq 的源码就知道了:

pub trait PartialEq<Rhs: ?Sized = Self> {
    /// This method tests for `self` and `other` values to be equal, and is used by `==`.
    #[must_use]
    #[stable(feature = "rust1", since = "1.0.0")]
    #[rustc_diagnostic_item = "cmp_partialeq_eq"]
    fn eq(&self, other: &Rhs) -> bool;
    /// 下面省略
}

可见,eq 方法接受的 selfother 本来就是引用类型,因此引用类型的 self 正中其下怀。

实际上,检查两个引用是否指向同一个对象的场景应当使用 std::ptr::eq 。其函数原型是:

pub fn eq<T: ?Sized>(a: *const T, b: *const T) -> bool {
    a == b
}

可见其原理是把运算对象当成了指针来比较,因此能够满足我们的需求。

知识共享许可协议
本作品采用知识共享署名-非商业性使用-相同方式共享 4.0 国际许可协议进行许可。
暂无评论

发送评论 编辑评论


				
|´・ω・)ノ
ヾ(≧∇≦*)ゝ
(☆ω☆)
(╯‵□′)╯︵┴─┴
 ̄﹃ ̄
(/ω\)
∠( ᐛ 」∠)_
(๑•̀ㅁ•́ฅ)
→_→
୧(๑•̀⌄•́๑)૭
٩(ˊᗜˋ*)و
(ノ°ο°)ノ
(´இ皿இ`)
⌇●﹏●⌇
(ฅ´ω`ฅ)
(╯°A°)╯︵○○○
φ( ̄∇ ̄o)
ヾ(´・ ・`。)ノ"
( ง ᵒ̌皿ᵒ̌)ง⁼³₌₃
(ó﹏ò。)
Σ(っ °Д °;)っ
( ,,´・ω・)ノ"(´っω・`。)
╮(╯▽╰)╭
o(*////▽////*)q
>﹏<
( ๑´•ω•) "(ㆆᴗㆆ)
😂
😀
😅
😊
🙂
🙃
😌
😍
😘
😜
😝
😏
😒
🙄
😳
😡
😔
😫
😱
😭
💩
👻
🙌
🖕
👍
👫
👬
👭
🌚
🌝
🙈
💊
😶
🙏
🍦
🍉
😣
Source: github.com/k4yt3x/flowerhd
颜文字
Emoji
小恐龙
花!
上一篇
下一篇