# 不定長類型
> [unsized-types.md](https://github.com/rust-lang/rust/blob/master/src/doc/book/unsized-types.md)
commit 6ba952020fbc91bad64be1ea0650bfba52e6aab4
大部分類型有一個特定的大小,以字節為單位,它們在編譯時是已知的。例如,一個`i32`是32位大,或者4個字節。然而,有些類型有益于表達,卻沒有一個定義的大小。它們叫做“不定長”或者“動態大小”類型。一個例子是`[T]`。這個類型代表一個特定數量`t`的序列。不過我們并不知道有多少,所以大小是未知的。
Rust知道幾個這樣的類型,不過它們有一些限制。這有三個:
1. 我們只能通過指針操作一個不定長類型的實例。`&[T]`剛好能正常工作,不過`[T]`不行。一個`&[T]`能正常工作,不過一個`[T]`不行。
1. 變量和參數不能擁有動態大小類型。
1. 只有一個`struct`的最后一個字段可能擁有一個動態大小類型;其它字段則不可以擁有動態大小類型。枚舉變量不可以用動態大小類型作為數據。
所以為什么這很重要?好吧,因為`[T]`只能用在一個指針之后,如果我們沒有對不定長類型的語言支持,它將不可能這么寫:
~~~
impl Foo for str {
~~~
或者
~~~
impl<T> Foo for [T] {
~~~
相反,你將不得不這么寫:
~~~
impl Foo for &str {
~~~
意味深長的是,這個實現將只能用于[引用](#),并且不能用于其它類型的指針。通過`impl for str`,所有指針,包括(在一些地方,這里會有bug需要修復)用戶自定義的智能指針,可以使用這個`impl`。
### `?Sized`
如果你想要寫一個接受動態大小類型的函數,你可以使用這個特殊的限制,`?Sized`:
~~~
struct Foo<T: ?Sized> {
f: T,
}
~~~
這個`?`,讀作“`T`可能是`Sized`的”,意味著這個限制是特殊的:它讓我們的匹配更寬松,而不是相反。這幾乎像每個`T`都隱式擁有`T: Sized`一樣,`?`放松了這個默認(限制)。
- 前言
- 貢獻者
- 1.介紹
- 2.準備
- 3.學習 Rust
- 3.1.猜猜看
- 3.2.哲學家就餐問題
- 3.3.其它語言中的 Rust
- 4.語法和語義
- 4.1.變量綁定
- 4.2.函數
- 4.3.原生類型
- 4.4.注釋
- 4.5.If語句
- 4.6.循環
- 4.7.所有權
- 4.8.引用和借用
- 4.9.生命周期
- 4.10.可變性
- 4.11.結構體
- 4.12.枚舉
- 4.13.匹配
- 4.14.模式
- 4.15.方法語法
- 4.16.Vectors
- 4.17.字符串
- 4.18.泛型
- 4.19.Traits
- 4.20.Drop
- 4.21.if let
- 4.22.trait 對象
- 4.23.閉包
- 4.24.通用函數調用語法
- 4.25.crate 和模塊
- 4.26.const和static
- 4.27.屬性
- 4.28.type別名
- 4.29.類型轉換
- 4.30.關聯類型
- 4.31.不定長類型
- 4.32.運算符和重載
- 4.33.Deref強制多態
- 4.34.宏
- 4.35.裸指針
- 4.36.不安全代碼
- 5.高效 Rust
- 5.1.棧和堆
- 5.2.測試
- 5.3.條件編譯
- 5.4.文檔
- 5.5.迭代器
- 5.6.并發
- 5.7.錯誤處理
- 5.8.選擇你的保證
- 5.9.外部函數接口
- 5.10.Borrow 和 AsRef
- 5.11.發布途徑
- 5.12.不使用標準庫
- 6.Rust 開發版
- 6.1.編譯器插件
- 6.2.內聯匯編
- 6.4.固有功能
- 6.5.語言項
- 6.6.鏈接進階
- 6.7.基準測試
- 6.8.裝箱語法和模式
- 6.9.切片模式
- 6.10.關聯常量
- 6.11.自定義內存分配器
- 7.詞匯表
- 8.語法索引
- 9.參考文獻
- 附錄:名詞中英文對照