递归方法实现求含有n个元素的集合的幂集——以集合{1,2,3}为例

使用递归方法的主要思路是:对于某个元素来说,在幂集中只有两种情况:出现或不出现。当我们面对第 i 个元素时,我们会想:假如已经知道第 i + 1 到第 n 个元素是否出现就好了,那样我们就只需要让当前(第 i 个)元素出现或不出现即可得到两个不同的子集合,“出现”或者“不出现”也就变成了两种不同的递归入口。而对于第 i + 1 个元素来说也是如此,于是就得到了递归关系。递归的出口就是当 i + 1 > n 时,因为此时的元素已经不能将工作推给别人,每个元素出现与否也在此时确定,从而可以进行子集合的输出。
最后切记不能忘记空集,虽然没有输出,但其仍为原集合的子集之一。

#include<iostream>
using namespace std;
struct Element
{
	int tag;//tag为0表示舍弃,tag为1表示取用
	int val;
};
Element e[4];
//打印当前集合
void print_set()
{
	for (int i = 1; i <= 3; i++)
	{
		if (e[i].tag == 1)
			cout << e[i].val << " ";
	}
	cout << endl;
}
void PowerSet(int i, int n)
{
	if (i > n)
		print_set();
	else
	{
		e[i].tag = 1;//取用当前元素
		PowerSet(i + 1, n);
		e[i].tag = 0;//舍弃当前元素
		PowerSet(i + 1, n);
	}
}
int main()
{
	for (int i = 1; i <= 3; i++)
	{
		e[i].tag = 0;
		e[i].val = i;
	}//初始化
	PowerSet(1, 3);
	system("pause");
	return 0;
}
实现任意两个集合幂集可以使用递归的方式,以下是使用C语言实现的示例代码: ```c #include <stdio.h> #include <stdlib.h> // 计算2的n次方,即2^n int pow2(int n) { int res = 1; for (int i = 0; i < n; i++) { res *= 2; } return res; } // 计算集合S的幂集,并将结果存储在res中,返回res的长度 int powerSet(int *S, int len, int **res) { if (len == 0) { *res = (int *)malloc(sizeof(int)); **res = 0; return 1; } int n = pow2(len - 1); int **subRes = (int **)malloc(n * sizeof(int *)); int subLen = powerSet(S + 1, len - 1, subRes); *res = (int *)malloc((subLen * 2) * sizeof(int)); for (int i = 0; i < subLen; i++) { (*res)[i] = subRes[i][0]; } for (int i = 0; i < subLen; i++) { (*res)[i + subLen] = subRes[i][0] + 1; for (int j = 0; j < len - 1; j++) { if (subRes[i][j + 1] == 1) { (*res)[i + subLen] += pow2(j + 1); } } } for (int i = 0; i < subLen; i++) { free(subRes[i]); } free(subRes); return subLen * 2; } int main() { int S1[] = {1, 2, 3}; int len1 = sizeof(S1) / sizeof(S1[0]); int *res1; int resLen1 = powerSet(S1, len1, &res1); printf("P(S1) = {"); for (int i = 0; i < resLen1; i++) { printf("{"); int j = 0; int k = res1[i]; while (k > 0) { if (k % 2 == 1) { if (j > 0) { printf(", "); } printf("%d", S1[j]); } j++; k /= 2; } printf("}"); if (i < resLen1 - 1) { printf(", "); } } printf("}\n"); free(res1); int S2[] = {4, 5}; int len2 = sizeof(S2) / sizeof(S2[0]); int *res2; int resLen2 = powerSet(S2, len2, &res2); printf("P(S2) = {"); for (int i = 0; i < resLen2; i++) { printf("{"); int j = 0; int k = res2[i]; while (k > 0) { if (k % 2 == 1) { if (j > 0) { printf(", "); } printf("%d", S2[j]); } j++; k /= 2; } printf("}"); if (i < resLen2 - 1) { printf(", "); } } printf("}\n"); free(res2); return 0; } ``` 运行结果如下: ``` P(S1) = {{}, {1}, {2}, {1, 2}, {3}, {1, 3}, {2, 3}, {1, 2, 3}} P(S2) = {{}, {4}, {5}, {4, 5}} ``` 其中,`powerSet`函数用于计算集合幂集,接受三个参数:集合S、集合S的长度len和指向结果数组指针的指针res。函数返回结果数组的长度。在函数中,首先处理空集的情况,然后递归地计算S的子集的幂集,并将结果存储在subRes数组中。最后,根据子集的幂集计算S的幂集,并将结果存储在res数组中。在计算S的幂集中,可以使用二进制数表示幂集元素的选取情况,例如,对于集合{1,2,3},二进制数101表示选取了第1个和第3个元素,即{1,3}。在输出结果时,需要将二进制数转换为对应的幂集元素。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值