C++20中的span容器及用法小結
一.span容器
span
是 C++20 中引入的一個新的標準容器,它用于表示連續(xù)的一段內(nèi)存區(qū)間,類似于一個輕量級的只讀數(shù)組容器。
span
是一個輕量級的非擁有式容器,它提供了對連續(xù)內(nèi)存的引用。 span
的主要用途是作為函數(shù)參數(shù),可以避免不必要的內(nèi)存拷貝,并且可以防止懸垂指針和空指針引用的問題。
它的定義在頭文件 <span>
中,并位于 std
命名空間中。span
包含了一個指向連續(xù)內(nèi)存區(qū)域的指針以及它所占用的大小,可以通過它來訪問這個內(nèi)存區(qū)域中的元素。
span
主要用于以下場景:
- 作為函數(shù)的參數(shù),用于指示函數(shù)需要處理的數(shù)據(jù)范圍;
- 作為類的成員變量,用于表示對象所管理的內(nèi)存區(qū)域;
- 作為數(shù)組的視圖,用于訪問數(shù)組的一部分
二.span的用法
下面是幾種 span 的用法示例:
1.將數(shù)組轉(zhuǎn)換為 span:
int arr[] = {1, 2, 3, 4, 5}; span<int> s(arr, 5);
這里將一個整型數(shù)組 arr
轉(zhuǎn)換為 span 類型,并使用數(shù)組首地址和元素個數(shù)作為參數(shù)。
2.使用 span 來遍歷一個容器:
vector<int> vec = {1, 2, 3, 4, 5}; for (auto&& x : span(vec)) { cout << x << " "; }
這里使用 span(vec)
來構造一個 span 對象,遍歷其中的元素并輸出。
3.使用 span 來獲取子序列:
int arr[] = {1, 2, 3, 4, 5}; span<int> s(arr, 5); auto s1 = s.subspan(1, 3);
這里將一個 span 對象 s
分割為從第 1 個元素開始,長度為 3 的子序列,并將結果存儲到 s1
中。
4.將 span 轉(zhuǎn)換為其他容器類型:
int arr[] = {1, 2, 3, 4, 5}; span<int> s(arr, 5); vector<int> vec(s.begin(), s.end());
這里使用 s.begin()
和 s.end()
將 span 對象 s
轉(zhuǎn)換為迭代器范圍,并使用這個迭代器范圍構造一個 vector 容器 vec
。
三.span的底層原理
下面為 span的簡化版源碼,用于展示其基本實現(xiàn):
template<typename T, std::size_t Extent = std::dynamic_extent> class span { public: // 定義迭代器類型 using iterator = T*; using const_iterator = const T*; // 構造函數(shù) constexpr span() noexcept : data_(nullptr), size_(0) {} constexpr span(T* ptr, std::size_t count) : data_(ptr), size_(count) {} template <std::size_t N> constexpr span(T(&arr)[N]) noexcept : data_(arr), size_(N) {} template <typename Container> constexpr span(Container& c) noexcept : data_(c.data()), size_(c.size()) {} // 拷貝構造函數(shù)和拷貝賦值運算符 constexpr span(const span& other) noexcept = default; span& operator=(const span& other) noexcept = default; // 訪問元素和迭代器操作 constexpr T* data() const noexcept { return data_; } constexpr std::size_t size() const noexcept { return size_; } constexpr bool empty() const noexcept { return size_ == 0; } constexpr T& operator[](std::size_t idx) const { return data_[idx]; } constexpr T& front() const { return data_[0]; } constexpr T& back() const { return data_[size_-1]; } constexpr iterator begin() const noexcept { return data_; } constexpr iterator end() const noexcept { return data_ + size_; } constexpr const_iterator cbegin() const noexcept { return data_; } constexpr const_iterator cend() const noexcept { return data_ + size_; } private: T* data_; // 元素指針 std::size_t size_; // 元素數(shù)量 };
具體實現(xiàn)方式是通過指針來引用連續(xù)的一段內(nèi)存,從而實現(xiàn) span 的基本功能。由于 span 沒有實際的內(nèi)存所有權,所以它不能擁有或釋放內(nèi)存。它只是提供了對現(xiàn)有內(nèi)存塊的訪問。
標準庫中的 span 還提供了一些其他的功能,比如對子區(qū)間的切片和子區(qū)間的迭代器等。實際的實現(xiàn)可能會更加復雜,但其基本的思想是一致的。
四.span 與 array ,vector ,數(shù)組指針 的區(qū)別
1. span 與 array ,vector的區(qū)別
span
是 C++20 中新增的一個輕量級容器,用于表示一段連續(xù)的內(nèi)存區(qū)域,它不負責管理內(nèi)存空間,也不會擁有所指向內(nèi)存的所有權,只是提供一種方便的方式來操作內(nèi)存區(qū)域,因此可以看做是一個只讀的“裸指針”。
與 array
和 vector
相比,span
的主要區(qū)別在于它不擁有自己的存儲空間,而是引用了另一個數(shù)組或容器的內(nèi)存空間。因此,當我們需要使用一個連續(xù)的內(nèi)存塊時,可以使用 span
來代替 array
或 vector
。
具體來說,array
是一個固定大小的數(shù)組容器,其大小在編譯時就確定了,不能動態(tài)改變。vector
是一個動態(tài)增長的數(shù)組容器,可以動態(tài)分配內(nèi)存,并在需要時擴大容量。而 span
是一個非擁有型的容器,可以看作是一個指向連續(xù)內(nèi)存區(qū)域的引用,可以指向任何類型的元素。
在使用方面,array
和 vector
可以用來存儲數(shù)據(jù),并通過下標或迭代器來訪問其中的元素;span
則更多地用來表示一段內(nèi)存區(qū)域,并提供類似于迭代器的操作來訪問其中的元素(就是 只讀),如 begin
、end
、rbegin
、rend
等。
總之,span
、array
和 vector
三者各有所長,可以根據(jù)實際需求來選擇使用。
2. span 與 數(shù)組指針的區(qū)別
在C++中,數(shù)組和指針是密不可分的,它們常常被一起使用。然而,數(shù)組和指針不是相同的東西,它們有自己的屬性和限制。同樣地,span
和指針也有很多區(qū)別,這里列舉幾點:
span
是一個封裝了數(shù)組指針和長度的輕量級容器,它提供了對數(shù)組的安全訪問。指針只是一個指向內(nèi)存位置的地址,沒有長度信息。因此,使用指針時需要顯式地傳遞長度信息,否則可能會導致緩沖區(qū)溢出等問題。
span
支持范圍操作,它可以使用STL中的算法和其他支持范圍操作的庫進行操作。指針只能通過指針運算和下標操作來訪問和操作數(shù)據(jù)。
span
是可傳遞性的,可以傳遞到函數(shù)中作為參數(shù),而指針不能。這是因為在函數(shù)中傳遞指針時,我們必須顯式地傳遞指針所指向的內(nèi)存塊的大小,否則函數(shù)無法確定內(nèi)存塊的大小。
span
是一個類模板,可以指定數(shù)據(jù)類型和長度類型。指針只能指向特定類型的數(shù)據(jù)。
總的來說,span
比指針更安全,更靈活,更易于使用,是一種更好的數(shù)組容器類型。
五.span的優(yōu)點
std::span
的主要優(yōu)點如下:
輕量級:
std::span
本身只是一個輕量級的非擁有式容器,沒有自己的內(nèi)存管理,因此可以在不分配內(nèi)存的情況下輕松地傳遞和操作數(shù)據(jù)。同時,std::span
的內(nèi)存布局與原始數(shù)組相同,因此不需要進行數(shù)據(jù)的復制或重排。安全性:
std::span
具有邊界檢查機制,可以避免訪問越界等錯誤,從而提高代碼的安全性。可組合性:
std::span
可以與其他容器類型進行組合,例如可以從std::vector
或std::array
中創(chuàng)建std::span
,或?qū)?std::span
轉(zhuǎn)換為std::vector
或std::array
。易于擴展:由于
std::span
只是一個非擁有式容器,因此可以輕松地將其用作接口的一部分,并以此擴展接口的功能。
總之,std::span
是一個非常實用的工具,可以方便地對數(shù)據(jù)進行訪問和處理,同時也可以提高代碼的可讀性、可維護性和安全性。
具體一些相關的信息見:std::span - cppreference.com
到此這篇關于C++20中的span容器的文章就介紹到這了,更多相關C++20中的span內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關文章希望大家以后多多支持腳本之家!
相關文章
Qt基礎開發(fā)之Qt多線程類QThread與Qt定時器類QTimer的詳細方法與實例
這篇文章主要介紹了Qt基礎開發(fā)之Qt多線程類QThread與Qt定時器類QTimer的詳細方法與實例,需要的朋友可以參考下2020-03-03C語言實現(xiàn)圖書管理系統(tǒng)(文件數(shù)據(jù)庫)
這篇文章主要為大家詳細介紹了C語言實現(xiàn)圖書管理系統(tǒng),文中示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下2022-03-03