C++系統學習之九:順序容器

来源:https://www.cnblogs.com/ZhouYong1991/archive/2018/09/17/9648067.html
-Advertisement-
Play Games

元素在順序容器中的順序與其加入容器時的位置相對應。關聯容器中元素的位置由元素相關聯的關鍵字值決定。所有容器類都共用公共的介面,不同容器按不同方式對其進行擴展。 一個容器就是一些特定類型對象的集合。順序容器為程式員提供了控制元素存儲和訪問順序的能力。 1. 順序容器概述 容器的兩種性能: 向容器中添加 ...


  元素在順序容器中的順序與其加入容器時的位置相對應。關聯容器中元素的位置由元素相關聯的關鍵字值決定。所有容器類都共用公共的介面,不同容器按不同方式對其進行擴展。

  一個容器就是一些特定類型對象的集合。順序容器為程式員提供了控制元素存儲和訪問順序的能力。

1. 順序容器概述

容器的兩種性能:

  • 向容器中添加或刪除元素的代價
  • 非順序訪問容器中元素的代價

順序容器類型:

如何確定使用哪種順序容器

NOTE:通常情況下,使用vector是最好的選擇。根據使用的需求對容器的哪種性能更加看重來選擇合適的容器。

2. 容器庫概覽

 容器類型上的操作有三個層次:

  • 某些操作是所有容器類型都提供的
  • 另外一些操作僅針對順序容器、關聯容器或無序容器
  • 還有一些操作只適用於一小部分容器

每個容器都定義在一個頭文件中,文件名與類型名相同。容器均定義為模板類。

以下容器操作是所有容器類型所共有的操作

2.1 迭代器

迭代器範圍

一個迭代器範圍由一對迭代器表示,begin指向容器的首元素,end指向容器的尾元素之後的位置。

迭代器範圍是一個左閉合區間,[begin, end)。

begin可以和end指向同一個位置,但是end不能指向begin之前的位置。

2.2 容器類型成員

處理迭代器之外還有反向迭代器,對反向迭代器進行++操作,會得到上一個元素。

通過value_type得到元素類型,通過reference或const_reference得到元素類型的一個引用。

2.3 標準庫array具有固定大小

與內置數組一樣,array的大小也是類型的一部分。定義array時,除了指定元素類型,還要指定容器大小。

array<int, 42>  arr;

其構造函數和數組差不多,列表初始化時,列表元素不要超過array的大小,可以少,array剩下的元素就進行值初始化。

array<int, 10> a={1,2,3};

剩下的7個元素值初始化為0;

此外還可以對array進行拷貝或對象賦值操作,但內置數組是不支持的

int a1[3]={0,1,2};
int a2[3]=a1;    //錯誤
array<int, 3> a3={0,1,2};
array<int, 3> a4=a3;    //正確

 拷貝或賦值要保證類型相同。

2.4 賦值和swap

註意array在初始化的時候可以進行列表初始化,但是賦值的時候不能將一個花括弧列表賦值給它。

NOTE:array賦值的時候,兩邊的運算對象必須具有相同的類型。

array<int, 3> a1 = { 0, 1, 2 };
array<int, 3> a2 = { 0 };
array<int, 3> a3 = { 2, 3, 4 };
a1 = a2;
a3 = { 0 };    //錯誤

對a3賦值的時候,a3的類型是array<int,3>,而右邊是先把{0}轉換成臨時變數array<int,1> tmp,然後再將tmp賦值給a3,這是tmp和a3類型不一致。

由於可能因為大小不一致導致類型不一致,array不支持容器的賦值運算assign。

NOTE:assign操作僅適合順序容器

賦值運算符要求左右兩邊的運算對象具有相同的類型,將右邊運算對象所有元素拷貝到左邊運算對象中。

swap

swap交換兩個相同類型容器的內容。swap操作並不會交換元素本身,知識交換兩個容器的內部數據結構(除array外)。這就意味著,指向容器的迭代器、引用和指針在swap操作之後都不會失效,仍指向swap操作之前的那些元素,只是這些元素屬於不同容器了。

swap兩個array會真正交換他們的元素。

有兩個版本的swap:成員函數版本以及非成員版本,統一使用非成員版本的swap。

2.5 容器大小操作

  • size:forward_list不支持size
  • empty
  • max_size

 2.6 關係運算符

每個容器類型都支持相等運算符(=和!=),除了無序容器外都支持關係運算符。關係運算符兩邊的運算對象必須是相同類型的容器,且必須保存相同類型的元素。

關係比較規則:

對元素進行逐對比較。大小相同、對應元素相等則相等;元素相等,大小不等,小的小於大的;都不相等,則比較第一個不相等的元素。

NOTE:只有當容器的元素定義了相應的比較運算符操作時,才可以使用關係運算符。

3. 順序容器操作

 順序容器和關聯容器的不同之處在於兩者組織元素的方式。這些不同之處直接關係到了元素如何存儲、訪問、添加以及刪除。接下來的操作都是順序容器所獨有的操作。

3.1 向順序容器中添加元素

新標準引入了三個新成員:emplace_front 、emplace和emplace_back,這些操作構造而不是拷貝元素。

當調用push或insert成員函數時,我們將元素類型的對象傳遞給它們,這些對象被拷貝到容器中。而當我們調用emplace成員函數時,則是將參數傳遞給元素類型的構造函數。

empalce函數的參數根據元素類型而變化,參數必須與元素類型的構造函數相匹配。

c.empalce("978-001",25,15.99);    //用三個參數直接構造對象
c.push_back("978-001",25,15.99);    //錯誤,沒有接受三個參數的push_back版本
c.push_back(Sales_data("978-001",25,15.99));    //正確

NOTE:emplace函數在容器中直接構造元素,而push_back則是創建臨時對象。

3.2 訪問元素

訪問成員函數返回的是引用

front、back、下標和at,其返回的都是引用。

如果我們使用auto變數來保存這些函數的返回值,並且希望使用此變數來改變元素的值,必須記得將變數定義為引用類型。

auto &v=c.back();    //獲得指向最後一個元素的引用
auto v=c.back();    //v不是一個引用,是c.back()的一個拷貝

 下標操作必須保證在範圍內操作

3.3 刪除元素

3.4 特殊的forward_list操作

forward_list並未定義insert、emplace和erase,而是定義了名為insert_after、emplace_after和erase_after的操作。

3.5 改變容器大小

list<int> ls(10,45);    //10個int,都是45
ls.resize(15);    //將5個值為0的元素添加到ls的末尾
ls.resize(25,-1);    //將10個值為-1的元素添加到ls的末尾
ls.resize(5);    //從ls末尾刪除20個元素

 3.6 容器操作可能使迭代器失效

由於向迭代器添加元素和從迭代器刪除元素的代碼可能會使迭代器失效,因此必須保證每次改變容器的操作之後都正確地重新定位迭代器。

4. vector對象是如何增長的

 每次都比實際所需空間多分配點空間,而不是每次都把所有元素移動到新空間。

管理容量的成員函數

capacity和size

size是指已經保存的元素的數目;而capacity則是在不分配新的記憶體空間的前提下它最多可以保存多少元素。

NOTE:只有當迫不得已時才可以分配新的記憶體空間。

5. 額外的string操作

 5.1 構造string的其他方法

string定義的這些額外操作要麼是提供string類和C風格字元數組之間的相互轉換,要麼是增加了允許我們用下標代替迭代器的版本。

substr操作

substr返回一個string,它是原始string的一部分或全部的拷貝,可以傳遞給substr一個可選的開始位置和計數值。

5.2 改變string的其他方法

 

string類型支持順序容器的賦值運算符以及assign、insert和erase操作。除此之外,還定義了額外的insert和erase版本。

s.insert(s.size(),5,'!');    //在s末尾插入5個感嘆號
s.erase(s.size()-5,5);    //從s刪除最後5個字元

const char *cp="Stately, plump Buck";
s.assign(cp,7);    //s=="Stately"
s.insrt(s.size(),cp+7);    //s=="Stately, plump Buck"

string s="some string", s2="some other string";
s.insert(0,s2);    //在s中位置0之前插入s2的拷貝
s.insert(0,s2,0,s2.size());

 5.3 string搜索操作

搜索是大小寫敏感的

逆向搜索

rfind成員函數搜索最後一個匹配,即子字元串最靠右的出現位置。

5.4 compare函數

5.5 數值轉換

6. 容器適配器

 除了順序容器外,標準庫還定義了三個順序容器適配器:stack、queue和priority_queue。容器、迭代器和函數都有適配器。

本質上,一個適配器是一種機制,能使某種事物的行為看起來像另一種事物一樣。

一個容器適配器接受一種已有的容器類型,使其行為看起來像一種不同的類型。

 


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 開發者的javascript造詣取決於對【動態】和【非同步】這兩個詞的理解水平。 一. 一道考察非同步知識的面試題 題目是這樣的,要求寫出下麵代碼的輸出: 如果沒有詳細鑽研過非同步隊列,答對的可能性很低。題目的考察點很明確,就是 中最核心的特點之一的【非同步】,瞭解了原理以後,你就會明白 中聲稱的“無阻塞” ...
  • 首發鏈接:https://bbs.huaweicloud.com/blogs/70f69ca4953111e89fc57ca23e93a89f 《一統江湖的大前端》系列是自己的前端學習筆記,旨在介紹javascript在非網頁開發領域的應用案例和發現各類好玩的js庫,不定期更新。如果你對前端的理解還 ...
  • 很精彩的一次內部分享,介紹了大部分的GC演算法理論知識,JVM博大精深,本篇文章只是結合本次內部分享總結的一些理論知識,如果有大佬有疑問,歡迎留言指出! Concurrent:併發,程式一邊運行一邊做GC Parallel:並行,一塊區域,一個人做清掃,需要100s,但是把區域分成兩塊,用兩個人掃,時 ...
  • 電腦端登錄公眾號管理後臺,【添加功能插件】開通客服功能,輸入"人工客服"接入客服熱線 底部有我的微信二維碼,如有問題,可加好友進行技術交流! ​ ​ ​ ​ ​ ​ ​ weixin-java-mp集成微信公眾號自帶客服功能代碼 增加TextBuilder.java文件 內容如下: public c ...
  • 博客園首頁是需要分享乾貨的地方,今天早上寫的《HRMS(人力資源管理系統)-從單機應用到SaaS應用-系統介紹》內容下架了,所以我就按照相關規定,只分享乾貨,我把之前寫完的架構設計相關知識的內容整理髮布上來。這次主要分享一下在架構設計過程中涉及的基礎知識,主要是涵蓋系統架構方法、架構模式及設計模式,... ...
  • 上周發佈的《2018,全新出發(全力推動實現住有所居》文章,其中記錄了個人在這5年過程中的成長和收穫,有幸認識了不少博客園的朋友,大家一起學習交流,在這個過程當中好多朋友提出SaaS系統如何設計,架構方面如何下手,在這5年的過程中我參與規劃設計了很多的SaaS系統其中有不少的坑和痛苦的經驗,特別是在... ...
  • RabbitMQ實戰教程(一) : 安裝及相關概念介紹 由於本人只在 安裝 服務 ,其他系統安裝暫時沒有涉及,如果有需要請自行搜索安裝教程. . . 1 . Windows 安裝 安裝需要先安裝 ,再安裝 第一步:安裝 ,由於 是用 編寫的,所以在安裝 之前要先安裝 下載地址: 下載最新版本即可,例 ...
  • 前言 此系列是針對springboot的啟動,旨在於和大家一起來看看springboot啟動的過程中到底做了一些什麼事。如果大家對springboot的源碼有所研究,可以挑些自己感興趣或者對自己有幫助的看;但是如果大家沒有研究過springboot的源碼,不知道springboot在啟動過程中做了些 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...