对于集合的交集,并集,差集的函数的理解

对于集合的交集,并集,差集的函数的理解

算法简介:

  • set_intersection // 求两个容器的交集(intersection==>交叉,交叉路口,十字路口,交集,交叉点)
  • set_union // 求两个容器的并集(union==>联盟,工会,联合,结合,并集)
  • set_difference // 求两个容器的差集(difference==>差异,差别,区别,差值
交集

功能:

  • 求两个容器的交集

函数原型:

  • set_intersection(iterator beg1, iterator end1, iterator beg2, iterator end2, iterator dest);
  • // beg1 容器1开始迭代器
    // end1 容器1结束迭代器
    // beg2 容器2开始迭代器
    // end2 容器2结束迭代器
    // dest 目标容器开始迭代器

需要注意的是:这两个集合都必须是有序序列,也就是都排过序的

#include"head.h"


int main()
{
	vector<int>a = { 1,3,5,6,4 ,7};
	sort(a.begin(), a.end(), [](int a, int b)->bool {return a < b; });//此处只相当于复习了一下lambda表达式,因为之前只学过,没用过,在这里先试试,其实就相当于默认从小到大的排序;
		vector<int>b = { 2,3,6,2,7,9 };
		sort(b.begin(), b.end(), [](int a, int b) {return a < b; });
		vector<int>c(100);
		//cout << c.size();
	auto it=	set_intersection(a.begin(), a.end(), b.begin(), b.end(), c.begin());
	for_each(c.begin(), it, [](int i) {cout << i << endl; });
	//3,6,7
		
}

lambda表达式

总结:

  • 求交集的两个集合必须是有序序列(必须是有序,否则一定会出错)求交集(以及并集差集)的时候其实很容易想到需要排序,如果是自己编写,只有排序才会加快时间复杂度,
  • 目标容器开辟空间需要从两个容器中求最小值(不是必要的,目的就是为了减少空间的浪费)
  • set_intersection的返回值时交集中最后一个元素的迭代器
set_union

功能描述:

  • 求两个集合的并集

函数原型:

set_union(iterator beg1, iterator end1, iterator beg2, iterator end2, iterator dest);

// end1 容器1结束迭代器
// beg2 容器2开始迭代器
// end2 容器2结束迭代器
// dest 目标容器开始迭代器

#include"head.h"


int main()
{
	vector<int>a = { 1,2,3,4,5,6 };
	vector<int>b = {0,1,3,5,6,9 };
	vector<int>c(60);

	auto it=set_union(a.begin(), a.end(), b.begin(), b.end(), c.begin());//事实证明必须要排序,否则一定会出错,错误提示无效参数的传递,不过上述数组有序,所以不用排序;
	for_each(c.begin(), it, [](int i) {cout << i << " "; });
		
}

总结

  • 求并集的两个集合必须为有序序列
  • 目标容器开辟空间需要两个容器相加(不是必须的,是为了节省空间)
  • set_union返回值时并集中最后一个元素的迭代器
set_difference

功能描述:

  • 求两个集合的差集

函数原型:

set_difference(iterator beg1, iterator end1, iterator beg2, iterator end2, iterator dest);

// 注意:两个集合必须是有序序列

// beg1 容器1开始迭代器
// end1 容器1结束迭代器
// beg2 容器2开始迭代器
// end2 容器2结束迭代器
// dest 目标容器开始迭代器

#include"head.h"


int main()
{
	vector<int>a = {1,7,5,3,6 };
	vector<int>b = {4,7,5,3 };
	vector<int>c(60);
	sort(a.begin(), a.end(), [](int a, int b) {return a < b; });//一定要排序
	sort(b.begin(), b.end(), [](int a, int b) {return a < b; });
	auto it=set_difference(a.begin(), a.end(), b.begin(), b.end(), c.begin());//相当于数学中集合a-b;
	for_each(c.begin(), it, [](int i) {cout << i << endl; });
}

**总结 **:

  • 求差集的两个集合必须是有序序列
  • 目标容器开辟空间需要从两个容器中取较大值(非必要,只是为了减少空间的浪费)
  • set_difference的返回值是差集中最后一个元素的位置的迭代器;
在C语言中,处理数据结构特别是集合(如数组、链表或哈希表等)的并集交集差集操作,可以使用多种方法,具体取决于数据存储的方式。以下是基本的思路: 1. **集合的表示**:一种常见的做法是将集合视为无序的元素列表。例如,可以使用数组或动态数组(如`int*`指针和大小)来存储整数元素。 2. **交集**: - 对于两个已排序的集合,可以使用双指针法遍历两个数组,找到第一个匹配的元素,并将其添加到结果数组中。遍历结束后,结果就是交集。 3. **并集**: - 可以创建一个新的空数组(或列表),然后将两个集合的所有元素都添加进去,避免重复即可。 4. **差集**: - 如果其中一个集合是另一个集合的子集,直接从大集合中移除小集合的元素即可得到差集。否则,需要分别计算两个集合并集再减去较小的那个集合。 5. **哈希表/集合**(如果使用更高级的数据结构): - 使用哈希表可以快速地进行查找和插入操作,使得上述操作更为高效,尤其是对于大量数据。 ```c #include <stdio.h> #include <stdlib.h> // 假设集合元素是整数,且用数组表示 void union_set(int *set1, int *set2, int n1, int m1, int n2, int *result, int *result_size) { // ... 实现并集... } void intersection_set(int *set1, int *set2, int n1, int m1, int n2, int *result, int *result_size) { // ... 实现交集... } void difference_set(int *set1, int *set2, int n1, int m1, int n2, int *result, int *result_size) { // ... 实现差集... } // 示例函数 void print_set(int *set, int size) { for (int i = 0; i < size; ++i) printf("%d ", set[i]); printf("\n"); } int main() { int set1[] = {1, 2, 3, 4, 5}; int set2[] = {4, 5, 6, 7}; int n1 = sizeof(set1) / sizeof(set1[0]); int n2 = sizeof(set2) / sizeof(set2[0]); int result[100]; // 假设最大并集长度不超过100 int result_size = 0; intersection_set(set1, set2, n1, n2, result, &result_size); print_set(result, result_size); union_set(set1, set2, n1, n2, result, &result_size); print_set(result, result_size); difference_set(set1, set2, n1, n2, result, &result_size); print_set(result, result_size); return 0; } ``` 注意:这只是一个简化版本的示例,实际实现会更复杂,包括错误检查和边界条件处理。同时,这个代码假设了输入集合没有重复元素。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值