欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

利用Rust實(shí)現(xiàn)一個(gè)簡單的Ping應(yīng)用

 更新時(shí)間:2022年12月06日 16:47:26   作者:qingwave  
這兩年Rust火的一塌糊涂,甚至都燒到了前端,再不學(xué)習(xí)怕是要落伍了。最近翻了翻文檔,寫了個(gè)簡單的Ping應(yīng)用練練手,感興趣的小伙伴可以了解一下

這兩年Rust火的一塌糊涂,甚至都燒到了前端,再不學(xué)習(xí)怕是要落伍了。最近翻了翻文檔,寫了個(gè)簡單的Ping應(yīng)用練練手,被所有權(quán)折騰的夠嗆,相比起Golang上手難度大很多,現(xiàn)將開發(fā)中的一些問題總結(jié)如下,所有源碼見ring

目標(biāo)

實(shí)現(xiàn)一個(gè)Ping,功能包含:

命令行解析

實(shí)現(xiàn)ICMP協(xié)議,pnet包中已經(jīng)包含了ICMP包定義,可以使用socket2庫發(fā)送

周期性發(fā)送Ping,通過多線程發(fā)送,再匯總結(jié)果

監(jiān)聽退出信號(hào)

命令行解析

系統(tǒng)庫std::env::args可以解析命令行參數(shù),但對(duì)于一些復(fù)雜的參數(shù)使用起來比較繁瑣,更推薦clap。利用clap的注解,通過結(jié)構(gòu)體定義命令行參數(shù)

/// ping but with rust, rust + ping -> ring
#[derive(Parser, Debug, Clone)] // Parser生成clap命令行解析方法
#[command(author, version, about, long_about = None)]
pub struct Args {
    /// Count of ping times
    #[arg(short, default_value_t = 4)] // short表示開啟短命名,默認(rèn)為第一個(gè)字母,可以指定;default_value_t設(shè)置默認(rèn)值
    count: u16,

    /// Ping packet size
    #[arg(short = 's', default_value_t = 64)]
    packet_size: usize,

    /// Ping ttl
    #[arg(short = 't', default_value_t = 64)]
    ttl: u32,

    /// Ping timeout seconds
    #[arg(short = 'w', default_value_t = 1)]
    timeout: u64,

    /// Ping interval duration milliseconds
    #[arg(short = 'i', default_value_t = 1000)]
    interval: u64,

    /// Ping destination, ip or domain
    #[arg(value_parser=Address::parse)] // 自定義解析
    destination: Address,
}

clap可以方便的指定參數(shù)命名、默認(rèn)值、解析方法等,運(yùn)行結(jié)果如下

?  ring git:(main) cargo run -- -h
   Compiling ring v0.1.0 (/home/i551329/work/ring)
    Finished dev [unoptimized + debuginfo] target(s) in 1.72s
     Running `target/debug/ring -h`
ping but with rust, rust + ping -> ring

Usage: ring [OPTIONS] <DESTINATION>

Arguments:
  <DESTINATION>  Ping destination, ip or domain

Options:
  -c <COUNT>            Count of ping times [default: 4]
  -s <PACKET_SIZE>      Ping packet size [default: 64]
  -t <TTL>              Ping ttl [default: 64]
  -w <TIMEOUT>          Ping timeout seconds [default: 1]
  -i <INTERVAL>         Ping interval duration milliseconds [default: 1000]
  -h, --help            Print help information
  -V, --version         Print version information

實(shí)現(xiàn)Ping

pnet中提供了ICMP包的定義,socket2可以將定義好的ICMP包發(fā)送給目標(biāo)IP,另一種實(shí)現(xiàn)是通過pnet_transport::transport_channel發(fā)送原始數(shù)據(jù)包,但需要過濾結(jié)果而且權(quán)限要求較高。

首先定義ICMP包

let mut buf = vec![0; self.config.packet_size];
let mut icmp = MutableEchoRequestPacket::new(&mut buf[..]).ok_or(RingError::InvalidBufferSize)?;
icmp.set_icmp_type(IcmpTypes::EchoRequest); // 設(shè)置為EchoRequest類型
icmp.set_icmp_code(IcmpCodes::NoCode);
icmp.set_sequence_number(self.config.sequence + seq_offset); // 序列號(hào)
icmp.set_identifier(self.config.id);
icmp.set_checksum(util::checksum(icmp.packet(), 1)); // 校驗(yàn)函數(shù)

通過socket2發(fā)送請(qǐng)求

let socket = Socket::new(Domain::IPV4, Type::DGRAM, Some(Protocol::ICMPV4))?;
let src = SocketAddr::new(net::IpAddr::V4(Ipv4Addr::UNSPECIFIED), 0);
socket.bind(&src.into())?; // 綁定源地址
socket.set_ttl(config.ttl)?;
socket.set_read_timeout(Some(Duration::from_secs(config.timeout)))?; // 超時(shí)配置
socket.set_write_timeout(Some(Duration::from_secs(config.timeout)))?;

???????// 發(fā)送
socket.send_to(icmp.packet_mut(), &self.dest.into())?;

最后處理相應(yīng),轉(zhuǎn)換成pnet中的EchoReplyPacket

let mut mem_buf = unsafe { &mut *(buf.as_mut_slice() as *mut [u8] as *mut [std::mem::MaybeUninit<u8>]) };
let (size, _) = self.socket.recv_from(&mut mem_buf)?;

???????// 轉(zhuǎn)換成EchoReply
let reply = EchoReplyPacket::new(&buf).ok_or(RingError::InvalidPacket)?;

至此,一次Ping請(qǐng)求完成。

周期性發(fā)送

Ping需要周期性的發(fā)送請(qǐng)求,比如秒秒請(qǐng)求一次,如果直接通過循環(huán)實(shí)現(xiàn),一次請(qǐng)求卡住將影響主流程,必須通過多線程來保證固定周期的發(fā)送。

發(fā)送請(qǐng)求

let send = Arc::new(AtomicU64::new(0)); // 統(tǒng)計(jì)發(fā)送次數(shù)
let _send = send.clone();
let this = Arc::new(self.clone());
let (sx, rx) = bounded(this.config.count as usize); // channel接受線程handler
thread::spawn(move || {
    for i in 0..this.config.count {
        let _this = this.clone();
        sx.send(thread::spawn(move || _this.ping(i))).unwrap(); // 線程中運(yùn)行ping,并將handler發(fā)送到channel中

        _send.fetch_add(1, Ordering::SeqCst); // 發(fā)送一次,send加1

        if i < this.config.count - 1 {
            thread::sleep(Duration::from_millis(this.config.interval));
        }
    }
    drop(sx); // 發(fā)送完成關(guān)閉channel
});
  • thread::spawn可以快速創(chuàng)建線程,但需要注意所有權(quán)的轉(zhuǎn)移,如果在線程內(nèi)部調(diào)用方法獲取變量,需要通過Arc原子引用計(jì)數(shù)
  • send變量用來統(tǒng)計(jì)發(fā)送數(shù),原子類型,并且用Arc包裹;this是當(dāng)前類的Arc克隆,會(huì)轉(zhuǎn)移到線程中
  • 第一個(gè)線程內(nèi)周期性調(diào)用ping(),并且其在單獨(dú)線程中運(yùn)行
  • 通過bounded來定義channel(類似Golang中的chan),用來處理結(jié)果,發(fā)送完成關(guān)閉

處理結(jié)果

let success = Arc::new(AtomicU64::new(0)); // 定義請(qǐng)求成功的請(qǐng)求
let _success = success.clone();
let (summary_s, summary_r) = bounded(1); // channel來判斷是否處理完成
thread::spawn(move || {
    for handle in rx.iter() {
        if let Some(res) = handle.join().ok() {
            if res.is_ok() {
                _success.fetch_add(1, Ordering::SeqCst); // 如果handler結(jié)果正常,success加1
            }
        }
    }
    summary_s.send(()).unwrap(); // 處理完成
});

第二個(gè)線程用來統(tǒng)計(jì)結(jié)果,channel通道取出ping線程的handler,如果返回正常則加1

處理信號(hào)

let stop = signal_notify()?; // 監(jiān)聽退出信號(hào)
select!(
    recv(stop) -> sig => {
        if let Some(s) = sig.ok() { // 收到退出信號(hào)
            println!("Receive signal {:?}", s);
        }
    },
    recv(summary_r) -> summary => { // 任務(wù)完成
        if let Some(e) = summary.err() {
            println!("Error on summary: {}", e);
        }
    },
);

通過select來處理信號(hào)(類似Golang中的select),到收到退出信號(hào)或者任務(wù)完成時(shí)繼續(xù)往下執(zhí)行。

信號(hào)處理

Golang中可以很方便的處理信號(hào),但在Rust中官方庫沒有提供類似功能,可以通過signal_hook與crossbeam_channel實(shí)現(xiàn)監(jiān)聽退出信號(hào)

fn signal_notify() -> std::io::Result<Receiver<i32>> {
    let (s, r) = bounded(1); // 定義channel,用來異步接受退出信號(hào)

    let mut signals = signal_hook::iterator::Signals::new(&[SIGINT, SIGTERM])?; // 創(chuàng)建信號(hào)

    thread::spawn(move || {
        for signal in signals.forever() { // 如果結(jié)果到信號(hào)發(fā)送到channel中
            s.send(signal).unwrap();
            break;
        }
    });

    Ok(r) // 返回接受channel
}

其他

很多吐槽人Golang的錯(cuò)誤處理,Rust也不遑多讓,不過提供了?語法糖,也可以配合anyhow與thiserror來簡化錯(cuò)誤處理。

驗(yàn)證

Ping域名/IP

ring git:(main)  cargo run -- www.baidu.com 

PING www.baidu.com(103.235.46.40)
64 bytes from 103.235.46.40: icmp_seq=1 ttl=64 time=255.85ms
64 bytes from 103.235.46.40: icmp_seq=2 ttl=64 time=254.17ms
64 bytes from 103.235.46.40: icmp_seq=3 ttl=64 time=255.41ms
64 bytes from 103.235.46.40: icmp_seq=4 ttl=64 time=256.50ms

--- www.baidu.com ping statistics ---
4 packets transmitted, 4 received, 0% packet loss, time 3257.921ms

測試退出信息,運(yùn)行中通過Ctrl+C中止

cargo run 8.8.8.8 -c 10

PING 8.8.8.8(8.8.8.8)
64 bytes from 8.8.8.8: icmp_seq=1 ttl=64 time=4.32ms
64 bytes from 8.8.8.8: icmp_seq=2 ttl=64 time=3.02ms
64 bytes from 8.8.8.8: icmp_seq=3 ttl=64 time=3.24ms
^CReceive signal 2

???????--- 8.8.8.8 ping statistics ---
3 packets transmitted, 3 received, 0% packet loss, time 2365.104ms

總結(jié)

Rust為了安全高效,通過引入所有權(quán)來解決GC問題,也帶來了許多不便,編程時(shí)必須要考慮到變量的聲明周期、借用等問題,所有語言都是在方便、性能、安全之間做權(quán)衡,要么程序員不方便,要么編譯器多做點(diǎn)功。換一個(gè)角度來說Bug總是不可避免的,在編譯階段出現(xiàn)總好過運(yùn)行階段。

到此這篇關(guān)于利用Rust實(shí)現(xiàn)一個(gè)簡單的Ping應(yīng)用的文章就介紹到這了,更多相關(guān)Rust實(shí)現(xiàn)Ping內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • Rust用宏實(shí)現(xiàn)參數(shù)可變的函數(shù)的實(shí)現(xiàn)示例

    Rust用宏實(shí)現(xiàn)參數(shù)可變的函數(shù)的實(shí)現(xiàn)示例

    本文主要介紹了Rust用宏實(shí)現(xiàn)參數(shù)可變的函數(shù)的實(shí)現(xiàn)示例,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2024-03-03
  • Rust日期與時(shí)間的操作方法

    Rust日期與時(shí)間的操作方法

    Rust的時(shí)間操作主要用到chrono庫,接下來我將簡單選一些常用的操作進(jìn)行介紹,感興趣的朋友跟隨小編一起看看吧
    2023-09-09
  • Rust中字符串類型&str和String的使用

    Rust中字符串類型&str和String的使用

    在Rust中,字符串是一種非常重要的數(shù)據(jù)類型,&str和String是Rust中兩種主要的字符串類型,本文主要介紹了Rust中字符串類型&str和String的使用,感興趣的可以了解一下
    2024-03-03
  • Rust 語言中的dyn 關(guān)鍵字及用途解析

    Rust 語言中的dyn 關(guān)鍵字及用途解析

    在Rust中,"dyn"關(guān)鍵字用于表示動(dòng)態(tài)分發(fā)(dynamic dispatch),它通常與trait對(duì)象一起使用,以實(shí)現(xiàn)運(yùn)行時(shí)多態(tài), 在Rust中,多態(tài)是通過trait和impl來實(shí)現(xiàn)的,這篇文章主要介紹了Rust 語言中的 dyn 關(guān)鍵字,需要的朋友可以參考下
    2024-03-03
  • rust中async/await的使用示例詳解

    rust中async/await的使用示例詳解

    在Rust中,async/await用于編寫異步代碼,使得異步操作更易于理解和編寫,通過使用await,在async函數(shù)或代碼塊中等待Future完成,而不會(huì)阻塞線程,允許同時(shí)執(zhí)行其他Future,這種機(jī)制簡化了異步編程的復(fù)雜性,使代碼更加直觀
    2024-10-10
  • 使用Rust實(shí)現(xiàn)日志記錄功能

    使用Rust實(shí)現(xiàn)日志記錄功能

    這篇文章主要為大家詳細(xì)介紹了使用Rust實(shí)現(xiàn)日志記錄功能的相關(guān)知識(shí),文中的示例代碼講解詳細(xì),具有一定的借鑒價(jià)值,有需要的可以參考一下
    2024-04-04
  • Rust 原始指針功能探索

    Rust 原始指針功能探索

    這篇文章主要為大家介紹了Rust 原始指針功能探索,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2023-10-10
  • Rust中的引用與借用舉例詳解

    Rust中的引用與借用舉例詳解

    這篇文章主要給大家介紹了關(guān)于Rust中引用與借用的相關(guān)資料,rust中借用和引用的附帶功效都一樣,就是都有生命周期,文中通過代碼介紹的非常詳細(xì),對(duì)大家學(xué)習(xí)或者使用Rust具有一定的參考價(jià)值,需要的朋友可以參考下
    2023-11-11
  • MacBook Pro安裝rust編程環(huán)境的過程

    MacBook Pro安裝rust編程環(huán)境的過程

    rustup是一個(gè)用于管理Rust版本和工具鏈的工具,這篇文章主要介紹了MacBook Pro安裝rust編程環(huán)境的過程,感興趣的朋友跟隨小編一起看看吧
    2024-02-02
  • 詳解thiserror庫在Rust中的使用

    詳解thiserror庫在Rust中的使用

    在編程中,錯(cuò)誤處理是一個(gè)至關(guān)重要的部分,在Rust中,我們經(jīng)常使用Result和Option類型來進(jìn)行錯(cuò)誤處理,但有時(shí),我們需要?jiǎng)?chuàng)建自定義的錯(cuò)誤類型,這就是thiserror庫發(fā)揮作用的地方,可以極大的簡化代碼,所以本文就給大家介紹一下如何使用thiserror
    2023-08-08

最新評(píng)論