酒店網(wǎng)站策劃書網(wǎng)站打開
泛型 Generics泛型詳解
使用泛型參數(shù),有一個(gè)先決條件,必需在使用前對(duì)其進(jìn)行聲明:
fn largest<T>(list: &[T]) -> T {
該泛型函數(shù)的作用是從列表中找出最大的值,其中列表中的元素類型為 T。首先 largest<T> 對(duì)泛型參數(shù) T 進(jìn)行了聲明,然后才在函數(shù)參數(shù)中進(jìn)行使用該泛型參數(shù) list: &[T] 。
下面是一個(gè)錯(cuò)誤的泛型函數(shù)的實(shí)現(xiàn):
fn largest<T>(list: &[T]) -> T {let mut largest = list[0];for &item in list.iter() {if item > largest {largest = item;}}largest
}fn main() {let number_list = vec![34, 50, 25, 100, 65];let result = largest(&number_list);println!("The largest number is {}", result);let char_list = vec!['y', 'm', 'a', 'q'];let result = largest(&char_list);println!("The largest char is {}", result);
}
運(yùn)行后報(bào)錯(cuò):
error[E0369]: binary operation `>` cannot be applied to type `T` // `>`操作符不能用于類型`T`--> src/main.rs:5:17|
5 | if item > largest {| ---- ^ ------- T| || T|
help: consider restricting type parameter `T` // 考慮對(duì)T進(jìn)行類型上的限制 :|
1 | fn largest<T: std::cmp::PartialOrd>(list: &[T]) -> T {| ++++++++++++++++++++++
因?yàn)?T 可以是任何類型,但不是所有的類型都能進(jìn)行比較,因此上面的錯(cuò)誤中,編譯器建議我們給 T 添加一個(gè)類型限制:使用 std::cmp::PartialOrd 特征(Trait)對(duì) T 進(jìn)行限制,特征在下一節(jié)會(huì)詳細(xì)介紹,現(xiàn)在你只要理解,該特征的目的就是讓類型實(shí)現(xiàn)可比較的功能。
結(jié)構(gòu)體中使用泛型
結(jié)構(gòu)體中的字段類型也可以用泛型來定義,下面代碼定義了一個(gè)坐標(biāo)點(diǎn) Point,它可以存放任何類型的坐標(biāo)值:
struct Point<T> {x: T,y: T,
}fn main() {let integer = Point { x: 5, y: 10 };let float = Point { x: 1.0, y: 4.0 };
}
這里有兩點(diǎn)需要特別的注意:
提前聲明,跟泛型函數(shù)定義類似,首先我們在使用泛型參數(shù)之前必需要進(jìn)行聲明 Point<T>,接著就可以在結(jié)構(gòu)體的字段類型中使用 T 來替代具體的類型
x 和 y 是相同的類型
第二點(diǎn)非常重要,如果使用不同的類型,那么它會(huì)導(dǎo)致下面代碼的報(bào)錯(cuò):
如果想讓 x 和 y 既能類型相同,又能類型不同,就需要使用不同的泛型參數(shù):
struct Point<T,U> {x: T,y: U,
}
fn main() {let p = Point{x: 1, y :1.1};
}
切記,所有的泛型參數(shù)都要提前聲明.
枚舉中使用泛型
提到枚舉類型,Option 永遠(yuǎn)是第一個(gè)應(yīng)該被想起來的,在之前的章節(jié)中,它也多次出現(xiàn):
enum Option<T> {Some(T),None,
}
Option<T> 是一個(gè)擁有泛型 T 的枚舉類型,它第一個(gè)成員是 Some(T),存放了一個(gè)類型為 T 的值。得益于泛型的引入,我們可以在任何一個(gè)需要返回值的函數(shù)中,去使用 Option<T> 枚舉類型來做為返回值,用于返回一個(gè)任意類型的值 Some(T),或者沒有值 None。
enum Result<T, E> {Ok(T),Err(E),
}
這個(gè)枚舉和 Option 一樣,主要用于函數(shù)返回值,與 Option 用于值的存在與否不同,Result 關(guān)注的主要是值的正確性。
如果函數(shù)正常運(yùn)行,則最后返回一個(gè) Ok(T),T 是函數(shù)具體的返回值類型,如果函數(shù)異常運(yùn)行,則返回一個(gè) Err(E),E 是錯(cuò)誤類型。例如打開一個(gè)文件:如果成功打開文件,則返回 Ok(std::fs::File),因此 T 對(duì)應(yīng)的是 std::fs::File 類型;而當(dāng)打開文件時(shí)出現(xiàn)問題時(shí),返回 Err(std::io::Error),E 對(duì)應(yīng)的就是 std::io::Error 類型。
方法中使用泛型
方法上也可以使用泛型:
struct Point<T> {x: T,y: T,
}impl<T> Point<T> {fn x(&self) -> &T {&self.x}
}
fn main() {let p = Point { x: 5, y: 10 };println!("p.x = {}", p.x());
}
使用泛型參數(shù)前,依然需要提前聲明:impl<T>,只有提前聲明了,我們才能在Point<T>中使用它,這樣 Rust 就知道 Point 的尖括號(hào)中的類型是泛型而不是具體類型。需要注意的是,這里的 Point<T> 不再是泛型聲明,而是一個(gè)完整的結(jié)構(gòu)體類型,因?yàn)槲覀兌x的結(jié)構(gòu)體就是 Point<T> 而不再是 Point。
除了結(jié)構(gòu)體中的泛型參數(shù),我們還能在該結(jié)構(gòu)體的方法中定義額外的泛型參數(shù),就跟泛型函數(shù)一樣:
struct Point<T, U> {x: T,y: U,
}impl<T, U> Point<T, U> {fn mixup<V, W>(self, other: Point<V, W>) -> Point<T, W> {Point {x: self.x,y: other.y,}}
}fn main() {let p1 = Point { x: 5, y: 10.4 };let p2 = Point { x: "Hello", y: 'c'};let p3 = p1.mixup(p2);println!("p3.x = {}, p3.y = {}", p3.x, p3.y);
}
這個(gè)例子中,T,U 是定義在結(jié)構(gòu)體 Point 上的泛型參數(shù),V,W 是單獨(dú)定義在方法 mixup 上的泛型參數(shù),它們并不沖突,說白了,你可以理解為,一個(gè)是結(jié)構(gòu)體泛型,一個(gè)是函數(shù)泛型。
為具體的泛型類型實(shí)現(xiàn)方法
對(duì)于 Point<T> 類型,你不僅能定義基于 T 的方法,還能針對(duì)特定的具體類型,進(jìn)行方法定義:
impl Point<f32> {fn distance_from_origin(&self) -> f32 {(self.x.powi(2) + self.y.powi(2)).sqrt()}
}
這段代碼意味著 Point<f32> 類型會(huì)有一個(gè)方法 distance_from_origin,而其他 T 不是 f32 類型的 Point<T> 實(shí)例則沒有定義此方法。這個(gè)方法計(jì)算點(diǎn)實(shí)例與坐標(biāo)(0.0, 0.0) 之間的距離,并使用了只能用于浮點(diǎn)型的數(shù)學(xué)運(yùn)算符。
const 泛型
const 泛型,也就是針對(duì)值的泛型,正好可以用于處理數(shù)組長度的問題:
fn display_array<T: std::fmt::Debug, const N: usize>(arr: [T; N]) {println!("{:?}", arr);
}
fn main() {let arr: [i32; 3] = [1, 2, 3];display_array(arr);let arr: [i32; 2] = [1, 2];display_array(arr);
}
如上所示,我們定義了一個(gè)類型為 [T; N] 的數(shù)組,其中 T 是一個(gè)基于類型的泛型參數(shù),這個(gè)和之前講的泛型沒有區(qū)別,而重點(diǎn)在于 N 這個(gè)泛型參數(shù),它是一個(gè)基于值的泛型參數(shù)!因?yàn)樗脕硖娲氖菙?shù)組的長度。
N 就是 const 泛型,定義的語法是 const N: usize,表示 const 泛型 N ,它基于的值類型是 usize。
在泛型參數(shù)之前,Rust 完全不適合復(fù)雜矩陣的運(yùn)算,自從有了 const 泛型,一切即將改變。
const 泛型表達(dá)式
假設(shè)我們某段代碼需要在內(nèi)存很小的平臺(tái)上工作,因此需要限制函數(shù)參數(shù)占用的內(nèi)存大小,此時(shí)就可以使用 const 泛型表達(dá)式來實(shí)現(xiàn):
// 目前只能在nightly版本下使用
#![allow(incomplete_features)]
#![feature(generic_const_exprs)]fn something<T>(val: T)
whereAssert<{ core::mem::size_of::<T>() < 768 }>: IsTrue,// ^-----------------------------^ 這里是一個(gè) const 表達(dá)式,換成其它的 const 表達(dá)式也可以
{//
}fn main() {something([0u8; 0]); // oksomething([0u8; 512]); // oksomething([0u8; 1024]); // 編譯錯(cuò)誤,數(shù)組長度是1024字節(jié),超過了768字節(jié)的參數(shù)長度限制
}// ---pub enum Assert<const CHECK: bool> {//
}pub trait IsTrue {//
}impl IsTrue for Assert<true> {//
}
const fn
@todo