練習:泛型 Logger
我們來設計一個簡單的記錄公用程式,使用 Logger
特徵搭配 log
方法。如果程式碼可能會記錄相關進度,就可以採用 &impl Logger
。在測試過程中,這可能會將訊息置於測試記錄檔中;而在實際版本中,則會將訊息傳送至記錄伺服器。
不過,下方的 StderrLogger
會記錄詳細程度不限的所有訊息。您的任務是編寫 VerbosityFilter
型別,忽略超出詳細程度上限的訊息。
以下是常見模式:結構體包裝一個特徵實作項目,並實作該相同特徵,在程序中加入行為。想一想,還有哪些其他類型的包裝函式可能在記錄公用程式中派上用場?
use std::fmt::Display;
pub trait Logger {
/// Log a message at the given verbosity level.
fn log(&self, verbosity: u8, message: impl Display);
}
struct StderrLogger;
impl Logger for StderrLogger {
fn log(&self, verbosity: u8, message: impl Display) {
eprintln!("verbosity={verbosity}: {message}");
}
}
fn do_things(logger: &impl Logger) {
logger.log(5, "FYI");
logger.log(2, "Uhoh");
}
// TODO: Define and implement `VerbosityFilter`.
fn main() {
let l = VerbosityFilter { max_verbosity: 3, inner: StderrLogger };
do_things(&l);
}