日韩性视频-久久久蜜桃-www中文字幕-在线中文字幕av-亚洲欧美一区二区三区四区-撸久久-香蕉视频一区-久久无码精品丰满人妻-国产高潮av-激情福利社-日韩av网址大全-国产精品久久999-日本五十路在线-性欧美在线-久久99精品波多结衣一区-男女午夜免费视频-黑人极品ⅴideos精品欧美棵-人人妻人人澡人人爽精品欧美一区-日韩一区在线看-欧美a级在线免费观看

歡迎訪問(wèn) 生活随笔!

生活随笔

當(dāng)前位置: 首頁(yè) > 编程语言 > c/c++ >内容正文

c/c++

C++语言中multiset的相关用法及扩展

發(fā)布時(shí)間:2023/12/20 c/c++ 32 豆豆
生活随笔 收集整理的這篇文章主要介紹了 C++语言中multiset的相关用法及扩展 小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,幫大家做個(gè)參考.

***************************************************

更多精彩,歡迎進(jìn)入:http://shop115376623.taobao.com

***************************************************


cpp語(yǔ)言中,multiset是<set>庫(kù)中一個(gè)非常有用的類型,它可以看成一個(gè)序列,插入一個(gè)數(shù),刪除一個(gè)數(shù)都能夠在O(logn)的時(shí)間內(nèi)完成,而且他能時(shí)刻保證序列中的數(shù)是有序的,而且序列中可以存在重復(fù)的數(shù)。

???? 我們通過(guò)一個(gè)程序來(lái)看如何使用multiset。

#include <string>
#include <iostream>
#include <set>
using?namespace?std;
void?main()

{
??? ?int?x;
???? scanf("%ld",&x);
???? multiset<int>?h;//建立一個(gè)multiset類型,變量名是h,h序列里面存的是int類型,初始h為空
???? while(x!=0)

? ? {
???? h.insert(x);//將x插入h中
? ? scanf("%ld",&x);

? ?}????
??? while(!h.empty())

? {

//?序列非空 h.empty()==true時(shí) 表示h已經(jīng)空了
???????? __typeof(h.begin())?c=h.begin();//c指向h序列中第一個(gè)元素的地址,第一個(gè)元素是最小的元素
???????? printf("%ld ",*c);//將地址c存的數(shù)據(jù)輸出
???????? h.erase(c);//從h序列中將c指向的元素刪除
???? }
}
對(duì)于輸入數(shù)據(jù)32 61 12 2 12 0,該程序的輸出是2 12 12 32 61。

我們可以看到,當(dāng)一個(gè)變量h被定義為multiset類型時(shí)。所有關(guān)于它的操作可以寫成如下格式:

h.函數(shù)名(形參);

當(dāng)要在h中插入一個(gè)數(shù)x時(shí),語(yǔ)法為h.insert(x);當(dāng)在h中刪除指針c指向的元素*c時(shí),語(yǔ)法為h.erase(c)。

注意,如果我們把h.erase(c)寫成h.erase(*c),那么該語(yǔ)句就會(huì)把h中所有和*c相等的元素都刪掉,大家要注意

如果要查找最大的元素并賦值給k,語(yǔ)法是int k=*(h.end()--),注意multiset類型的尾地址存的內(nèi)容是空的。

如果要想知道當(dāng)前序列中比k大的元素最小的是多少,那么可以這樣 int p=*(h.upper_bound(k)),其中h.upper_bound(k)表示比k大的最小的數(shù)的地址。

不光是int類型,multiset還可以存儲(chǔ)其他的類型諸如 string類型,結(jié)構(gòu)(struct或class)類型。而我們一般在編程當(dāng)中遇到的問(wèn)題經(jīng)常用到多關(guān)鍵字的類型,即struct或class。例如下面的例子:

struct?rec

{
????int?x,y;
};
multiset<rec>h;
以上的代碼是沒(méi)有任何用處的,因?yàn)閙ultiset并不知道如何去比較一個(gè)自定義的多關(guān)鍵字類型。怎么辦呢?我們可以定義multiset里面rec類型變量之間的小于關(guān)系的含義(這里以x為第一關(guān)鍵字為例),具體過(guò)程如下:

我們定義一個(gè)比較類cmp,cmp內(nèi)部的operator函數(shù)的作用是比較rec類型a和b的大小(以x為第一關(guān)鍵字,y為第二關(guān)鍵字):

struct?cmp

{
????bool?operator()(const?rec&a,const?rec&b)

? ? {
????????return?a.x<b.x||a.x==b.x&&a.y<b.y;
????}
};

?然后我們將語(yǔ)句"multiset<rec>h;” 改成"multiset<rec,cmp>h;"這樣以后,我們就告訴了序列h如何去比較里面的元素(這種方法屬于重載運(yùn)算符,在編程當(dāng)中經(jīng)常用到,這里就不詳細(xì)介紹了)

此時(shí)rec以及multiset的定義部分完整代碼可參考如下:

struct?rec{
????int?x,y;
};
struct?cmp{
????bool?operator()(const?rec&a,const?rec&b){
????????return?a.x<b.x||a.x==b.x&&a.y<b.y;
????}
};
multiset<rec,cmp>h;
通過(guò)以上代碼,我們就能建立一個(gè)集合h使得該集合能夠存儲(chǔ)和排序多關(guān)鍵字類型

我們來(lái)看一個(gè)小應(yīng)用:求從一個(gè)點(diǎn)到另一個(gè)點(diǎn)的最短路長(zhǎng)度,邊都是正權(quán)。

正權(quán)邊的最短路問(wèn)題可以用dijkstra算法來(lái)解決,而優(yōu)化dijkstra算法可以用heap。這里我們來(lái)看如何用multiset實(shí)現(xiàn)dijkstra+heap。

以下代碼省去了輸入輸出和圖的建立。我們光看求最短路的部分。注意,這里的多關(guān)鍵字類型名稱還是rec,multiset集合的名稱還是h;多關(guān)鍵字類型rec中,x是第一關(guān)鍵字,y是第二關(guān)鍵字,y代表圖中點(diǎn)的編號(hào),而x則代表當(dāng)前點(diǎn)y與源點(diǎn)的最短距離。

d[0]=0;//源點(diǎn)是0
reca;
a.x=0;//第一關(guān)鍵字x表示距離
a.y=0;//第二關(guān)鍵字y表示點(diǎn)的編號(hào)
h.insert(a);//將a插入序列中
while(!h.empty()){//h集合中的元素是否為空
??????__typeof(h.begin())?c=h.begin();
????rect=(*c);//取最小值
????h.erase(c);//將最小值刪去
????for(inti=tail[t.y];i;i=next[i]){
????????intj=p[i];//枚舉和t.y相鄰的點(diǎn)
????????reca;//建立一個(gè)結(jié)構(gòu)類變量a
????????if(d[j]==-1){//d[j]==-1表示j還沒(méi)有被訪問(wèn)
????????????d[j]=t.x+w[i];//w[i]表示邊i的邊權(quán)
????????????a.x=d[j];
????????????a.y=j;//將j的相關(guān)信息保存在rec類型a中
????????????h.insert(a);
????????}elseif(d[j]>t.x+w[i]){//最短路算法的松弛操作
????????????a.x=d[j];
????????????a.y=j;//將j在序列中的信息存儲(chǔ)到a中
????????????c=h.upper_bound(a);//找到序列h中a之后的元素的地址
????????????c--;//地址減一就是a所在的地址
????????????h.erase(c);//刪掉a
????????????a.x=t.x+w[i];
????????????d[j]=a.x;//更新最短路的值
????????????h.insert(a);//插入
????????}
????}
}
有了multiset類型,我們就不用再去寫平衡樹一類的東西了,從而大大降低了編程復(fù)雜度

總結(jié)

以上是生活随笔為你收集整理的C++语言中multiset的相关用法及扩展的全部?jī)?nèi)容,希望文章能夠幫你解決所遇到的問(wèn)題。

如果覺(jué)得生活随笔網(wǎng)站內(nèi)容還不錯(cuò),歡迎將生活随笔推薦給好友。