产生一个int数组,长度为100,并向其中随机插入1-100,并且不能重复

写在前面

前天去面试了,给出的笔试中有这样的一道算法题,产生一个int数组,长度为100,并向其中随机插入1-100,并且不能重复

当时,脑子一热,也没想那么多,就用集合实现了一下,经面试官提醒,发现还有更好的方式来实现。

代码

首先看一下这样一段代码

 1 namespace Wolfy.RandomDemo
 2 {
 3     class Program
 4     {
 5         static void Main(string[] args)
 6         {
 7             List<int> lst = new List<int>();
 8             Random r = new Random();
 9             while (true)
10             {
11                 int temp = r.Next(1, 101);
12                 if (lst.Count == 100)
13                 {
14                     break;
15                 }
16                 if (!lst.Contains(temp))
17                 {
18                     lst.Add(temp);
19                 }
20             }
21             for (int i = 0; i < lst.Count; i++)
22             {
23                 Console.WriteLine(lst[i]);
24             }
25             Console.Read();
26         }
27     }
28 }

虽然上面的代码,实现题目的要求,但是如果是1到100万或者更大,这样的每次判断是否包含这样的一个数,势必会影响到性能。

网上找到一种更好的实现方式:

(1)把N个数放到容器A(int数组)中.

(2)从N个数中随机取出1个数放入容器B(int数组)中.

(3)把容器A中最后一个数与随机抽取的数对调 或者 把容器A中最后一个数覆盖随机抽取出来的数.

(4)这时从容器A(假设N个数,索引0 到 索引N-2)之间随机取一个数.再放入容器B中,重复此步骤.

说明:也就是第二次是从容器A中 第一个元素到倒数第二个元素 中随机取一个数.

这种好处是,随机数所取范围逐步缩小,而且杜绝了大数据时集合执行删除操作时产生的瓶颈.

代码实现

 1 namespace Wolfy.RandomDemo
 2 {
 3     class Program
 4     {
 5         static void Main(string[] args)
 6         {
 7             int[] result = GetRandom(100);
 8             for (int i = 0; i < result.Length; i++)
 9             {
10                 Console.WriteLine(result[i]);
11             }
12             Console.WriteLine("over:" + result.Length);
13             Console.Read();
14         }
15         /// <summary>
16         /// 获得无重复随机数组
17         /// </summary>
18         /// <param name="n">上限n</param>
19         /// <returns>返回随机数组</returns>
20         static int[] GetRandom(int n)
21         {
22             //容器A和B
23             int[] arryA = new int[n];
24             int[] arryB = new int[n];
25             //填充容器a
26             for (int i = 0; i < arryA.Length; i++)
27             {
28                 arryA[i] = i + 1;
29             }
30             //随机对象
31             Random r = new Random();
32             //最后一个元素的索引 如n=100,end=99
33             int end = n - 1;
34             for (int i = 0; i < n; i++)
35             {
36                 //生成随机数 因为随机的是索引 所以从0到100取,end=100
37                 //一个大于等于 minValue 且小于 maxValue 的 32 位带符号整数,即:返回的值范围包括 minValue 但不包括 maxValue。
38                 //如果 minValue 等于 maxValue,则返回 minValue
39                 //
40                 int minValue = 0;
41                 int maxValue = end + 1;
42                 int ranIndex = r.Next(minValue, maxValue);
43                 //把随机数放在容器B中
44                 arryB[i] = arryA[ranIndex];
45                 //用最后一个元素覆盖取出的元素
46                 arryA[ranIndex] = arryA[end];
47                 //缩减随机数生成的范围
48                 end--;
49             }
50             //返回生成的随机数组
51             return arryB;
52         }
53     }
54 }

总结

实现方式有很多种,但是如果能用高效的方式就用高效的方式实现。这种生成无重复的随机数,可以在运用在抽奖系统中。

时间: 2024-12-16 14:11:41

产生一个int数组,长度为100,并向其中随机插入1-100,并且不能重复的相关文章

黑马程序员——java--将一个int数组转换成一个字符串

将一个int 数组转换成字符串 并输出在控制台上 //将一个int 数组转换成一个字符串 public class IntToStringDemo { public static void main(String[] args) { // TODO Auto-generated method stub //定义一个int类型的数组 int[] i ={4,5,8,6,5,8,7,4,5}; //调用自定义方法将int数组的方法转换成字符串 toStringMethod(i); } private

Random 产生一个int数组,长度为100,并向其中随机插入1-100,并且不能重复。

做法1: List<int> list = new List<int>(); Random rand = new Random();  while (list.Count < 100)             {                 int number = rand.Next(1, 101);//>=1,<101 if (!list.Contains(number))//如果list中已经含有这个数,则不插入 { list.Add(number);

Java -- 给定一个int数组,拼接出最大数值

public class ZhiJieTiaoDong { /* 给定一个数组:组合成最大数值 */ public String szpj(int[] args){ if(null == args || args.length == 0){ return ""; } StringBuilder stringBuilder = new StringBuilder(); int[] yxsz = compareSZ(args); for (int i = args.length - 1;

怎么把一个int数组转化为char型数组??

/* 234 Press any key to continue */ #include <stdio.h> int main() { int i,num = 234,n; char ch,s[10]; for(n = 0; num; ++n) { s[n] = num % 10 + '0'; num /= 10; } s[n] = '\0'; for(i = 0; i < n / 2; ++i) { ch = s[i]; s[i] = s[n - 1 - i]; s[n - 1 - i

查看一个int数组里边的每个数字出现过几次

public void aa() { int[] a = { 1, 2, 3, 4, 5, 4, 3, 2, 1 }; Hashtable ht = new Hashtable(); for (int i = 0; i < a.Length; i++) { if (ht.ContainsKey(a[i])) { ht.Add(a[i], (int)ht[i] + 1); } else { ht.Add(a[i], 1); } } string str = ""; foreach

signal num一个int数组,里面数值都是成对出现,只有一个是单独的,找出单独者。360ms

public class Solution { public int singleNumber(int[] nums) { int ret = nums[0]; for (int i = 1; i < nums.length; i++) { ret ^= nums[i]; } return ret; } }

有一个有序数组,要求将一个新输入的数插入到数组中并保证插入新数后,数组仍有序。

因为数组在初始条件下是有序的,而题目要求插入新数后数组仍有序.因此,首先应该在数组中找到插入位置,然后将该位置原有的数以及该位置后面的数都依次后移一个位置,为新插入的数值空出一个位置. #include<stdio.h> int main() { int a[11] = {2, 4, 8, 15, 20, 36, 49, 51, 78, 96}; int temp1, temp2, number, end, i, j; printf("The original arry is:\n&

返回一个整数数组最大子数组的和

开始时间:14:00 结束时间:17:   20 要求: 1,程序可以处理1000个元素: 2,每个元素是int32类型的: 3,整型数组有正数也有负数: 4,可以求出所有子数组的和的最大值,时间复杂度为O(n) 5,小组成员:于芳娜,张素颖 设计思路 1, 如果不考虑时间复杂度,我们可以枚举出所有子数组并求出他们的和.不过非常进憾的是,由于长度为n的数组有O(n^2)个子数组,而且求一个长度为n的数组的和的时间复杂度为O(n). 因此这种思路的时间是O(n^3). 2, 很容易理解,当我们加上

返回一个整数数组最大子数和。(新)

一.要求: (1)输入一个整形数组,数组里有正数也有负数. (2)数组中连续的一个或多个整数组成一个子数组,每个子数组都有一个和. (3)如果数组A[0]--A[j-1]首尾相邻,允许A[i-1],......A[n-1], A[0]--A[j-1]之和最大. 二.设计思想: 如果数组首尾相连,那么每一个元素都可以当成是开头即从A[n]开始,到A[n-1]结束,这样的话,就相当用剪子在一开始的数组中的元素的前一位剪开.这样的话,可以建立新的三个数组,前两个数组分别盛放剪开的两部分,第三个数组则是