注意:
eg. sort()
)while (!que.empty()) // 有clear()功能
{que.pop();
}
1 emplace 與 push 的區(qū)別1.直接傳入對(duì)象(int, double 或者 構(gòu)造好了的對(duì)象)
//假設(shè)棧內(nèi)的數(shù)據(jù)類型是data
class data {int a;
int b;
public:
data(int x, int y):a(x), b(y) {}
};
//push
data d(1,2);
stackS;
S.push(d) 或 S.emplace(d);
2.在傳入時(shí)候構(gòu)造對(duì)象
S.push(data(1,2));
S.emplce(data(1,2));
3.emplace可以直接傳入構(gòu)造對(duì)象需要的元素,然后自己調(diào)用其構(gòu)造函數(shù)!
S.emplace(1,2)
意思是,emplace這樣接受新對(duì)象的時(shí)候,自己會(huì)調(diào)用其構(gòu)造函數(shù)生成對(duì)象然后放在容器內(nèi)(比如這里傳入了1,2,它則會(huì)自動(dòng)調(diào)用一次data(1,2)),而push,只能讓其構(gòu)造函數(shù)構(gòu)造好了對(duì)象之后,再使用拷貝構(gòu)造函數(shù)!相當(dāng)于emplace直接把原料拿進(jìn)家,造了一個(gè)。而push是造好了之后,再?gòu)?fù)制到自己家里,多了復(fù)制這一步。所以emplace相對(duì)于push,使用第三種方法會(huì)更節(jié)省內(nèi)存。
注意:
emplace_back(type) 對(duì)應(yīng) push_back(type)
emplace(i, type) 對(duì)應(yīng)于 insert(type, i)
emplace_front(type) 對(duì)應(yīng)于 push_front()
棧中進(jìn)入數(shù)據(jù)稱為 ——入棧 s.push()
棧中彈出數(shù)據(jù)稱為 ——出棧 s.pop()
//構(gòu)造函數(shù)
stacks;? ? ? //stack采用類模板實(shí)現(xiàn), stack對(duì)象的默認(rèn)構(gòu)造形式
stacks1(s); ?//拷貝構(gòu)造函數(shù)
//賦值操作
s=s1;? ? //重載等號(hào)操作符
//數(shù)據(jù)存取
s.push(40);? ? ?//向棧頂添加元素
s.pop();? ? ? ? //從棧頂移除第一個(gè)元素
s.top() ? ? //返回棧頂元素值
//大小操作
s.empty() ? ? ? //判斷堆棧是否為空
s.size()? ? ? ? //返回棧的大?。ㄔ貍€(gè)數(shù))
#include//棧容器常用接口
void test01()
{//特點(diǎn):符合先進(jìn)后出數(shù)據(jù)結(jié)構(gòu)
stacks;
//向棧中添加元素,叫做 壓棧 入棧
s.push(10);
s.push(20);
s.push(30);
s.push(40);
cout<< "棧的大小為:"<< s.size()<< endl;
//只要棧不為空,查看棧頂,并且執(zhí)行出棧操作
while (!s.empty()) {//輸出棧頂元素
cout<< "棧頂元素為: "<< s.top()<< endl;
//彈出棧頂元素
s.pop();
}
cout<< "棧的大小為:"<< s.size()<< endl;
}
int main() {test01();
system("pause");
return 0;
}
二、queue (隊(duì)列)(先進(jìn)先出、【頭部刪除、尾部插入】、不許遍歷)
1 queue 基本概念[kju:]隊(duì)列中進(jìn)數(shù)據(jù)稱為 — 入隊(duì) q.push()
隊(duì)列中出數(shù)據(jù)稱為 — 出隊(duì) q.pop()
2 queue 常用接口(構(gòu)造函數(shù)、賦值操作、數(shù)據(jù)存儲(chǔ)、empty、size)//構(gòu)造函數(shù)
queueq; ? ? //queue采用類模板實(shí)現(xiàn),queue對(duì)象的默認(rèn)構(gòu)造形式
queueq1(q); //拷貝構(gòu)造函數(shù)
//賦值操作
q=q1; ? ? ? //重載等號(hào)操作符
//數(shù)據(jù)存取
q.push(p1); ? ? ? ? //往隊(duì)尾添加元素
q.pop(); ? ? ? ? //從隊(duì)頭移除第一個(gè)元素
q.back(); ? ? ? //返回最后一個(gè)元素值
q.front() ? ? ? //返回第一個(gè)元素值
//大小操作
q.empty() ? ? ? ? //判斷堆棧是否為空
q.size() ? ? ? ? //返回棧的大小
#include#include
class Person
{public:
Person(string name, int age)
{this->m_Name = name;
this->m_Age = age;
}
string m_Name;
int m_Age;
};
//隊(duì)列 Queue
void test01() {//創(chuàng)建隊(duì)列
queueq;
//準(zhǔn)備數(shù)據(jù)
Person p1("唐僧", 30);
Person p2("孫悟空", 1000);
Person p3("豬八戒", 900);
Person p4("沙僧", 800);
//向隊(duì)列中添加元素 入隊(duì)操作
q.push(p1);
q.push(p2);
q.push(p3);
q.push(p4);
//隊(duì)列不提供迭代器,更不支持隨機(jī)訪問(wèn)
while (!q.empty()) {//輸出隊(duì)頭元素
cout<< "隊(duì)頭元素-- 姓名: "<< q.front().m_Name
<< " 年齡: "<< q.front().m_Age<< endl;
cout<< "隊(duì)尾元素-- 姓名: "<< q.back().m_Name
<< " 年齡: "<< q.back().m_Age<< endl;
cout<< endl;
//彈出隊(duì)頭元素
q.pop();
}
cout<< "隊(duì)列大小為:"<< q.size()<< endl;
}
int main() {test01();
system("pause");
return 0;
}
3 priority_queue
3.1 用pair<int,int>建立優(yōu)先隊(duì)列(小根堆)priority_queue,vector>,greater>>pq;
1.pq.top().first;
2.pq.top().second;
三、list(最常用)(雙向迭代器、單向鏈表/雙向循環(huán)鏈表、可以遍歷)
1 list基本概念(插入刪除效率高,隨機(jī)存儲(chǔ)不行)(1)鏈表的組成:鏈表由一系列結(jié)點(diǎn)組成
(2)結(jié)點(diǎn)的組成:一個(gè)是存儲(chǔ)數(shù)據(jù)元素的數(shù)據(jù)域,一個(gè)是存儲(chǔ)下一個(gè)結(jié)點(diǎn)地址的指針域
(3)list的優(yōu)點(diǎn):
(4)list的缺點(diǎn):
(5)List有一個(gè)重要的性質(zhì):
總結(jié):STL中List和vector是兩個(gè)最常被使用的容器,各有優(yōu)缺點(diǎn)
listL1;? ? ? ? //list采用類模板實(shí)現(xiàn),對(duì)象的默認(rèn)構(gòu)造形式:
listL2(L1.begin(), L1.end());? //構(gòu)造函數(shù)將[beg, end)區(qū)間中的元素拷貝給本身。
listL4(5, 10); ? //構(gòu)造函數(shù)將5個(gè)10拷貝給本身
listL3(L2); ? //拷貝構(gòu)造函數(shù)。
//list容器的構(gòu)造
void printList(const list& L) { //只讀迭代器
for (list::const_iterator it = L.begin(); it != L.end(); it++) {cout<< *it<< " ";
}
cout<< endl;
}
void test01()
{//創(chuàng)建list容器
listL1;//默認(rèn)構(gòu)造
//添加數(shù)據(jù)
L1.push_back(10);
L1.push_back(20);
L1.push_back(30);
L1.push_back(40);
printList(L1);
//區(qū)間方式
listL2(L1.begin(), L1.end());
printList(L2);
//拷貝構(gòu)造
listL3(L2);
printList(L3);
//n個(gè)elem
listL4(10, 1000);
printList(L4);
}
int main() {test01();
system("pause");
return 0;
}
3 list 賦值和交換(operator=、assign、swap)//assign 賦值函數(shù)
L3.assign(L2.begin(), L2.end()); //將[beg, end)區(qū)間中的數(shù)據(jù)拷貝賦值給本身。
L4.assign(10, 100);? ? //將n個(gè)elem拷貝賦值給本身。
//operator= 重載
L2 = L1;? ? ? ? //重載等號(hào)操作符
//swap 互換函數(shù)
L1.swap(L2);? ? ? ? //將L1與L2的元素互換。
4 list 大小操作 (size、emply、resize) (無(wú)容量capacity)//(無(wú)容量capacity)
L1.size() //返回容器中元素的個(gè)數(shù)
L1.empty() //判斷容器是否為空
//resize 重新指定容器的長(zhǎng)度(大?。?L1.resize(10); //10個(gè)0
//重新指定容器的長(zhǎng)度為10,若容器變長(zhǎng),則以默認(rèn)值0填充新位置。
//如果容器變短,則末尾超出容器長(zhǎng)度的元素被刪除。
L1.resize(10,1000); //10個(gè)1000
//重新指定容器的長(zhǎng)度為10,若容器變長(zhǎng),則以1000值填充新位置。
//如果容器變短,則末尾超出容器長(zhǎng)度的元素被刪除。
5 list 插入和刪除 (insert、erase、remove(移除指定數(shù)據(jù))、clear)(迭代器找位置)//it表示某個(gè)迭代器,n為整數(shù)。該函數(shù)的功能是將it迭代器前進(jìn)或后退n個(gè)位置。
//n為正數(shù)時(shí)表示將it右移(前進(jìn))n個(gè)位置,n為負(fù)數(shù)時(shí)表示將it左移(后退)n個(gè)位置
#includeusing namespace std;
advance(it, n); // 要判斷 n 是否越界 (雙向循環(huán)鏈表也要防止越界)
Data.insert(it, tem2); //插入數(shù)據(jù)
Data.erase(it); //刪除數(shù)據(jù)
//兩端插入、刪除操作
L.push_back(10);???? //在容器尾部加入一個(gè)元素
L.pop_back();??? //刪除容器中最后一個(gè)元素
L.push_front(100);?? //在容器開(kāi)頭插入一個(gè)元素
L.pop_front();??? //從容器開(kāi)頭移除第一個(gè)元素
//insert 指定位置插入
list::iterator it;
L.insert(it, 10);? //在迭代器位置插10元素,返回新數(shù)據(jù)的位置。
L.insert(it, 2, 10); //在迭代器位置插入2個(gè)10數(shù)據(jù),無(wú)返回值。
L.insert(it, L2.begin(), L2.end());??//在迭代器位置插入[beg,end)區(qū)間的數(shù)據(jù),無(wú)返回值。
//erase 指定位置刪除
L.erase(L.begin(), L.end());?//刪除[beg,end)區(qū)間的數(shù)據(jù),返回下一個(gè)數(shù)據(jù)的位置。
L.erase(it);? ? //刪除迭代器位置的數(shù)據(jù),返回下一個(gè)數(shù)據(jù)的位置。
//remove 移除指定數(shù)據(jù)(特有)
L.remove(100);???? //刪除容器中所有與100值匹配的元素。
//clear 清除所有元素
L.clear();????? //移除容器的所有數(shù)據(jù)
6 list 數(shù)據(jù)存?。╢ront、back、不支持[ ]和at的方式)(迭代器找位置)//不支持[ ]和at的方式
L.front()? ? ? ? //返回第一個(gè)元素。
L.back()? ? ? ? //返回最后一個(gè)元素。
//只支持it++、it-- ,不支持 it = it + 1;
//雙向迭代器,不可以跳躍訪問(wèn),即使是+1
list::iterator it = L.begin();
int a = *it; //返回迭代器所在的位置
//數(shù)據(jù)存取
void test01()
{listL1;
L1.push_back(10);
L1.push_back(20);
L1.push_back(30);
L1.push_back(40);
//cout<< L1.at(0)<< endl;//錯(cuò)誤 不支持at訪問(wèn)數(shù)據(jù)
//cout<< L1[0]<< endl; //錯(cuò)誤 不支持[]方式訪問(wèn)數(shù)據(jù)
//不支持隨機(jī)訪問(wèn)迭代器
cout<< "第一個(gè)元素為: "<< L1.front()<< endl;
cout<< "最后一個(gè)元素為: "<< L1.back()<< endl;
//list容器的迭代器是雙向迭代器,不支持隨機(jī)訪問(wèn)
list::iterator it = L1.begin();
it++;//只能++,--
//it = it + 1;//錯(cuò)誤,不可以跳躍訪問(wèn),即使是+1
}
int main() {test01();
system("pause");
return 0;
}
7 list 反轉(zhuǎn)和排序(L.reverse()(反轉(zhuǎn))、L.sort()(排序,鏈表內(nèi)部提供))L.reverse(); ????//反轉(zhuǎn)鏈表
L.sort(); ???? //鏈表排序//默認(rèn)排序規(guī)則從小到大
void printList(const list& L) {for (list::const_iterator it = L.begin(); it != L.end(); it++) {cout<< *it<< " ";
}
cout<< endl;
}
//回調(diào)函數(shù)
bool myCompare(int val1, int val2)
{//降序,就讓第一個(gè)數(shù)大于第二個(gè)數(shù)
return val1 >val2;
}
//list容器反轉(zhuǎn)和排序
//反轉(zhuǎn)
void test01()
{listL;
L.push_back(90);
L.push_back(30);
L.push_back(20);
L.push_back(70);
cout<< "反轉(zhuǎn)前: "<< endl;
printList(L);
//反轉(zhuǎn)容器的元素
L.reverse();
cout<< "反轉(zhuǎn)后: "<< endl;
printList(L);
}
//排序
void test02()
{listL;
L.push_back(90);
L.push_back(30);
L.push_back(20);
L.push_back(70);
cout<< "排序前: "<< endl;
printList(L);
//所有不支持隨機(jī)訪問(wèn)迭代器的容器,不可以用標(biāo)準(zhǔn)算法
//不支持隨機(jī)訪問(wèn)迭代器的容器,內(nèi)部會(huì)提供對(duì)應(yīng)一些算法
L.sort();//默認(rèn)排序規(guī)則從小到大
cout<< "排序后:"<< endl;
printList(L);
L.sort(myCompare); //寫一個(gè)回調(diào)函數(shù),從大到小
printList(L);
}
int main() {test01();
test02();
system("pause");
return 0;
}
8、C++ STL 中l(wèi)ist是雙向循環(huán)鏈表,雙向可以理解,有兩個(gè)指針域,指向前一結(jié)點(diǎn)和指向后一結(jié)點(diǎn),雙向可以實(shí)現(xiàn)從末尾結(jié)點(diǎn)到頭結(jié)點(diǎn)的遍歷,但循環(huán)實(shí)現(xiàn)什么功能?
#include#includeint main()
{listli;
for (int i = 0; i< 5; ++i)
{li.push_back(i);
}
list::iterator it = li.end();
cout<< *(--it); //輸出4
cout<< *(++it); //若為循環(huán)鏈表,不是應(yīng)該回到頭結(jié)點(diǎn)嗎?實(shí)際輸出錯(cuò)誤!
//若為循環(huán)鏈表,那end()是指向哪里?
return 0;
}
一個(gè)node結(jié)構(gòu)自己是不知道自己是list中的第幾個(gè)(沒(méi)有儲(chǔ)存相應(yīng)的信息)。但是,最末一個(gè)node,它的后指針是指向鏈表的終結(jié)記號(hào),然后終結(jié)記號(hào)的node也有一個(gè)指針,才指向list的第一個(gè)node。所以,++it指向的是終結(jié)記號(hào),上面是沒(méi)有數(shù)據(jù)的,當(dāng)然輸出錯(cuò)誤。
雙向的意思是:你可以在首端加入新的數(shù)據(jù)node,也可以在末端加入新的數(shù)據(jù)node,但不表示你可以無(wú)限循環(huán)的遍歷它。另外,List模板,不建議你使用iterator(迭代器),因?yàn)槊恳粋€(gè)node都不知道自己是第幾個(gè)node,如果你使用迭代器指定你要訪問(wèn)第n個(gè)node的數(shù)據(jù),它總是從首元素開(kāi)始一個(gè)個(gè)數(shù)到第n,然后才返回?cái)?shù)據(jù)給你。最好把鏈表當(dāng)作動(dòng)態(tài)的雙端數(shù)組(deque)來(lái)使用,只訪問(wèn)或者增刪頭端或者尾端的數(shù)據(jù),這樣速度快
你是否還在尋找穩(wěn)定的海外服務(wù)器提供商?創(chuàng)新互聯(lián)www.cdcxhl.cn海外機(jī)房具備T級(jí)流量清洗系統(tǒng)配攻擊溯源,準(zhǔn)確流量調(diào)度確保服務(wù)器高可用性,企業(yè)級(jí)服務(wù)器適合批量采購(gòu),新人活動(dòng)首月15元起,快前往官網(wǎng)查看詳情吧