浅谈set与multiset的运用 ʕ •ᴥ•ʔ

今天遇到一道题需要multiset来写的题 就顺便学习了一下它的用法

异:set和multiset会根据特定的排序原则将元素排序。两者不同之处在于,multisets允许元素重复,而set不允许重复。

1 构造、拷贝、析构

操作

效果

set c

产生一个空的set/multiset,不含任何元素

set c(op)

以op为排序准则,产生一个空的set/multiset

set c1(c2)

产生某个set/multiset的副本,所有元素都被拷贝

set c(beg,end)

以区间[beg,end)内的所有元素产生一个set/multiset

set c(beg,end, op)

以op为排序准则,区间[beg,end)内的元素产生一个set/multiset

c.~set()

销毁所有元素,释放内存

set<Elem>

产生一个set,以(operator <)为排序准则

set<Elem,0p>

产生一个set,以op为排序准则

其实这一部分我看的不是很懂。。。

 2 非变动性操作

操作

效果

c.size()

返回当前的元素数量

c.empty ()

判断大小是否为零,等同于0 == size(),效率更高

c.max_size()

返回能容纳的元素最大数量

c1 == c2

判断c1是否等于c2

c1 != c2

判断c1是否不等于c2(等同于!(c1==c2))

c1 < c2

判断c1是否小于c2

c1 > c2

判断c1是否大于c2

c1 <= c2

判断c1是否小于等于c2(等同于!(c2<c1))

c1 >= c2

判断c1是否大于等于c2 (等同于!(c1<c2)) d

这一部分很有用!

3 特殊的搜寻函数

 sets和multisets在元素快速搜寻方面做了优化设计,提供了特殊的搜寻函数,所以应优先使用这些搜寻函数,可获得对数复杂度,而非STL的线性复杂度。比如在1000个元素搜寻,对数复杂度平均十次,而线性复杂度平均需要500次。

操作

效果

count (elem)

返回元素值为elem的个数

find(elem)

返回元素值为elem的第一个元素,如果没有返回end()

lower _bound(elem)

返回元素值为elem的第一个可安插位置,也就是元素值 >= elem的第一个元素位置

upper _bound (elem)

返回元素值为elem的最后一个可安插位置,也就是元素值 > elem 的第一个元素位置

equal_range (elem)

返回elem可安插的第一个位置和最后一个位置,也就是元素值==elem的区间

4 赋值

操作

效果

c1 = c2

将c2的元素全部给c1

c1.swap(c2)

将c1和c2 的元素互换

swap(c1,c2)

同上,全局函数

偷过来这些你是不是已经看不下去啦?再坚持一下~ 因为重点就要来啦!!

5 迭代器相关函数

 sets和multisets的迭代器是双向迭代器,对迭代器操作而言,所有的元素都被视为常数,可以确保你不会人为改变元素值,从而打乱既定顺序,所以无法调用变动性算法,如remove()。

操作

效果

c.begin()

返回一个随机存取迭代器,指向第一个元素

c.end()

返回一个随机存取迭代器,指向最后一个元素的下一个位置

c.rbegin()

返回一个逆向迭代器,指向逆向迭代的第一个元素

c.rend()

返回一个逆向迭代器,指向逆向迭代的最后一个元素的下一个位置

其实 begin()返回的是数组中第一个元素,rbegin返回的数组中最后的一个元素

end() rend()同理 只不过返回的都是下一个位置 所以一般他们返回的值相同

6.安插和删除元素

必须保证参数有效,迭代器必须指向有效位置,序列起点不能位于终点之后,不能从空容器删除元素。

操作

效果

c.insert(elem)

插入一个elem副本,返回新元素位置,无论插入成功与否。

c.insert(pos, elem)

安插一个elem元素副本,返回新元素位置,pos为收索起点,提升插入速度。

c.insert(beg,end)

将区间[beg,end)所有的元素安插到c,无返回值。

c.erase(elem)

删除与elem相等的所有元素,返回被移除的元素个数。

c.erase(pos)

移除迭代器pos所指位置元素,无返回值。

c.erase(beg,end)

移除区间[beg,end)所有元素,无返回值。

c.clear()

移除所有元素,将容器清空

光说没用 上手~

#include <bits/stdc++.h>
using namespace std;
int l[300005], r[300005];
multiset<string> ms;
int main()
{
	multiset<string>::iterator it;
	ms.insert("abc");
	ms.insert("1233");
	ms.insert("1111");
	ms.insert("aaa");
	ms.insert("1233");
	ms.insert("bbb");
	for(it=ms.begin();it!=ms.end();it++)
	{
		cout<<*it<<endl;
	}
	cout<<endl<<"集合大小"<<ms.size()<<endl; 
	it=ms.find("1233");
	if(it!=ms.end())
	{
		cout<<*it<<endl;
	}
	else
	cout<<"not found"<<endl;
	it=ms.find("43");
	if(it!=ms.end())
	cout<<*it<<endl;
	else
	cout<<"not found"<<endl;
	int n=ms.erase("1233");
	cout<<"共删除:"<<n<<endl;
	for(it=ms.begin();it!=ms.end();it++)
	cout<<*it<<endl;
	return 0;
}

输出的结果:

1111
1233
1233
aaa
abc
bbb

集合大小6
1233
not found
共删除:2
1111
aaa
abc
bbb

需要注意的点:

1.是以中序遍历去遍历整个集合的,在插入的时候自动调整

2.遍历的时候需要判断一下集合是否为空;

3.插入的数默认从小到大排序 set<int>::iterator it; 

4.如果要从大到小的话 set<int>::reverse_iterator rit;

猜你喜欢

转载自blog.csdn.net/henucm/article/details/82081885