随机生成和为S的N个正整数——投影法

本文介绍了一种高效有趣的算法——投影法,用于生成和为S的N个正整数。通过随机生成N-1个不同数、排序后计算两两差值得到最终结果。该方法的时间复杂度为O(N*logN),适用于数据量不大时优化效率。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

随机生成和为S的N个正整数——投影法

     随机生成和为SN个正整数有很多种解法。下面讲解一种比较高效且比较有趣味性的解法——投影法。

     以生成和为204个数为例,可以先生成随机生成020之间的三个数字再排序,假设得到了4718。然后在X-Y数轴上画出这三个数,如下图:




    然后将这些数值投影到Y轴上,可得下图:


      由图很容易看出ABBCCDDE这四段的长度和肯定为20。因此ABBCCDDE这四段的长度即和为204个数,这4个数分别为43112

     这种方法只要随机生成N - 1个不同数,然后排序后计算两两差值就可以得到和为SN个正整数,因此效率还是比较高的。下面给出完整代码:

  1. #include <cstdio>  
  2. #include <ctime>  
  3. #include <set>  
  4. #include <algorithm>  
  5. using namespace std;  
  6. //在[s, e)区间上随机取n个数并存放到a[]中  
  7. void GetRandomNum(int *a, int n, int s, int e)  
  8. {  
  9.     std::set<int> set_a;  
  10.     srand(time(NULL));  
  11.     for (int i = e - n; i < e; i++)  
  12.     {  
  13.         int num = (rand() % i) + s;  
  14.         if (set_a.find(num) == set_a.end())  
  15.             set_a.insert(num);  
  16.         else  
  17.             set_a.insert(i);  
  18.     }  
  19.     i = 0;  
  20.     std::set<int>::iterator pos;  
  21.     for (pos = set_a.begin(); pos != set_a.end(); pos++)  
  22.         a[i++] = *pos;  
  23. }  
  24. int main()  
  25. {  
  26.     const int NSUM = 20;  
  27.     const int NCOUNT = 4;  
  28.       
  29.     printf("           生成和为%d的%d个数 \n", NSUM, NCOUNT); 
  30.       
  31.     int a[NCOUNT];  
  32.       
  33.     GetRandNumberInRange(a, NCOUNT - 1, 0, NSUM);  
  34.     sort(a, a + NCOUNT - 1);  
  35.     a[NCOUNT - 1] = NSUM;  
  36.   
  37.     printf("  已经生成和为%d的%d个数: \n", NSUM, NCOUNT);  
  38.     printf("%d ", a[0]);  
  39.     for (int i = 1; i < NCOUNT; i++)  
  40.         printf("%d ", a[i] - a[i - 1]);  
  41.     putchar('\n');  
  42.     return 0;  
  43. }  

运算结果如下图所示:

 

     这种“投影法”能有效解决随机生成和为SN个正整数,其算法本质是通过“投影”得到各数据之间的长度差,而且这些长度差之和即投影线段的总长度显然会等于最大数据的值减去最小数据的值

 

下面分析下算法的时间复杂度:

      算法分为生成随机的N-1个数+排序+遍历共费时O(N * logN) +O(N * logN) + O(N),整体时间复杂度为O(N * logN)

      算法的最主要费时操作在排序上,如果数据量不是太大,使用基数排序可以将排序操作的时间复杂度降低到O(N)

      其次在生成随机的N-1个数时,虽然只要调用rand()随机函数N-1次,但由于使用了set来做数据存储的容器,因此每次插入数据前的查找要费时O(logN),插入新数据时也要费时O(logN),可以改用hast_set来进一步提高效率。


 参考:http://blog.youkuaiyun.com/morewindows/article/details/8439393


评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值