采用概率算法,估计整数子集1~n的大小

在估计一个集合的势(大小)时,如果不要求很高的精确度,却要求较快的运算效率。我们推荐使用概率算法来求解。

设X是具有n个元素的集合,我们有回放地随机,均匀和独立地从X中选取元素,设k是出现第1次重复之前所选出的元素数目,则当n足够大时,k的期望值趋近为β√n,

这里β=√(π/2),利用此结论可以得出估计|X|的概率算法:n=2k2

 

算法伪代码如下:

SetCount (X)
SetCount (X) {
              k ← 0; S ← Ф;
              a ← uniform(X);
              do {
                     k++;
                     S ← S∪{a}; a ← uniform(X);
              } while (a ∉S)
              return 2k2/π;
         }
 

C语言描述如下:

#include<stdio.h>
#include<math.h>
#include<stdlib.h> 
#include<time.h>
#define N 100000
a[N]={0};
void main()
{
  unsigned int a[N];
  unsigned int i,j,n,k=0;
  unsigned int uniform;
 srand( (unsigned)time(NULL));
 printf(" 输入集合大小:");
 scanf("%d",&n);

 for(i=0;i<n;i++)
 {uniform=rand()*rand()%n;
  a[i]=uniform;
     for(j=0;j<i;j++)
     {
      if(a[j]==a[i])  goto  result;
     }
 }

result: k=i;
printf("计算结果为%5.0f, 真实值为%d\n",2*k*k/3.14,n);
}

 

执行结果如下:

结论:

1:此方法适用于快速计算集合的势(大小),不适用于精确计算

2:本文执行结果为作者筛选后的结果。在执行过程中还是会出现一些不好的结果,但此概率计数不失为一个好方法。

 

转载于:https://www.cnblogs.com/yuzeren48/archive/2012/10/24/2737673.html

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
好的,我来为您解答。 这是一个比较典型的回溯算法问题,我们可以通过枚举所有的可能情况来找到所有子集。 具体的思路如下: 1. 首先定义一个递归函数,它的参数包括当前的数字、当前已选择的数字集合和所有子集的列表。 2. 对于当前数字,有两种选择:加入当前已选择的数字集合或不加入。如果加入,就将当前数字加入集合,并将集合保存到所有子集的列表中;如果不加入,就直接递归到下一个数字。 3. 递归调用函数,传入下一个数字和更新后的数字集合。 下面是具体的代码实现: ```python def subsets(n): results = [] def backtrack(start, subset): results.append(subset[:]) for i in range(start, n + 1): subset.append(i) backtrack(i + 1, subset) subset.pop() backtrack(1, []) return results ``` 其中,n表示给定的正整数,results是保存所有子集的列表。 函数backtrack是递归函数,它的参数包括当前数字start和当前已选择的数字集合subset。 在backtrack函数内部,首先将当前已选择的数字集合保存到结果列表中(注意,这里要使用subset的副本,而不是直接将subset添加到结果列表中,否则会保存错误的结果)。 然后,从当前数字开始,逐个尝试加入数字,如果加入,就递归调用函数,传入下一个数字和更新后的数字集合;如果不加入,就直接递归到下一个数字。 最后,我们可以通过调用subsets函数来获取所有子集,例如: ```python n = 3 results = subsets(n) for subset in results: print(subset) ``` 输出结果如下: ``` [] [1] [1, 2] [1, 2, 3] [1, 3] [2] [2, 3] [3] ``` 注意,这里的输出结果中包括空集和原集合本身,这是符合子集的定义的。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值