非重复随机序列的高效生成(C语言)

非重复随机序列的高效生成(C语言)

996.icu LICENSE

  • 问题描述
  • 思想+程序

阅读之前注意:

本文阅读建议用时:14min
本文阅读结构如下表:

项目下属项目测试用例数量
问题描述0
思想+程序1

问题描述

  1. 有一个长度为N的序列,现在要你把它全部打乱顺序,生成一个随机的序列

思想+程序

思路1:

  1. 生成随机数,要交换的序号=随机数%长度
  2. 判断要交换的序号是否已经被交换过,若交换过则跳回步骤1重新生成随机数,若没有交换过则交换后再跳回步骤1,直至所有的序号都被交换过

思路2:

  1. 第 i 次(i 从0开始且每次递增1)生成随机数,要交换的序号=随机数%(长度- i )
  2. 交换序号 i 和 步骤1中“要交换的序号”,直至 i 递增到了序列末尾

通常情况下我们会想到思路1,但当N非常大时,生成的随机数%长度 得到的序号很有可能是已经交换过的序号,导致效率很低。
而思路2通过巧妙的固定规律交换,即序号 i 是有规律递增的,而另一个要交换的序号是序列中序号 i 之后的任意一个随机序号,同样实现了随机性,且保证了时间复杂度为O(n),十分高效!
同样是交换,秘诀在于交换的法则。

我们用C语言实现了思路2
程序参考以下代码:

#include<stdio.h>
#include<stdlib.h>
#define N 10

void swap(int *a, int i, int j)
{
	int tmp = a[i];
	a[i] = a[j];
	a[j] = tmp;
}

void main()
{
	int i = 0;
	int a[N] = { 0 };
	for (i = 0; i < N; i++)
		a[i] = i + 1;
	for (i = 0; i < N; i++)//交换a[i]和a[i]后面的随机序号
		swap(a, i, i + rand() % (N - i));
	for (i = 0; i < N; i++)
		printf("%d ", a[i]);
	system("pause");
}

可以看到代码部分也十分简洁1

如果本文对你有帮助,不如请我一罐可乐吧 🍼
在这里插入图片描述


  1. 问题源于一次面试,同时感谢前辈的点拨. ↩︎

  • 1
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值