典型算法及应用——关于产生不重复随机数的算法

我们不得不承认这样一个事实:那就是尽管在高级程序语言设计中包含了类似于Random产生随机数之类的方法,但是它产生的随机数并不能满足我们日常的需要,因为它可能重复——设想一下,电子化抽取试题的原理就是根据预定产生的题目数量产生果敢若干个对应的随机数,然后将匹配的试题抽取、排序并打印在试卷上。但是在同一次考试时候不允许同一题目出现重复(尽管这样的概率很低,但是我们绝对不允许这样做!)。所以避免产生重复随机数的方法(产生“真正的随机数”)成了我们必须研究的话题。今天本文就讨论一下。

产生不重复随机数的方法有很多——我倾向于“筛选法”(当然,这个方法是我在学习随机数时候突然想出来的,可能之前就有许多人早就有了这种思想)。 所谓“筛选法”就是根据要产生随机数指定的范围(起始数必须小于终止数),将这些数全部装入一个数组,然后利用系统随机函数(比如 Random )随机产生一个下标,将这个下标对应的数值返回并删除对应的这个数,直到这个数组为空。

下面给出VB.NET和C#对应的代码:

(VB.NET)

Public   NotInheritable   Class TureRandom
    
Private nums As New ArrayList
    
Public Sub New(ByVal startnum As IntegerByVal endnum As Integer)
        
If (startnum >= endnum) Then
            
Throw New Exception("对不起,起始数字必须小于结尾数字!")
        
Else
            
For i As Integer = startnum To endnum
                nums.Add(i)
            
Next
        
End If
    
End Sub


    
Public Function GetNum() As Integer
        
If (nums.Count <= 0Then
            
Throw New Exception("对不起,指定范围的随机数全部产生过了。")
        
Else
            
Randomize()
            
Dim index As Integer = Rnd() * nums.Count
            
Dim returnnum As Integer = CType(nums(index), Integer)
            nums.RemoveAt(index)
            
Return returnnum
        
End If
    
End Function

End Class

(C#)

public   sealed   class  TureRandom
{
    
private ArrayList nums=new ArrayList();

    
public TureRandom (int startnum, int endnum)
   
{
        
if (startnum >= endnum) 
            
throw new Exception("对不起,起始数字必须小于结尾数字!")
        
else
           
for (int i=startnum; i<=endnum;++i)
                nums.Add(i);
       
    }


    public 
int GetNum()
   
{
        
if (nums.Count <= 0) Then
            
throw new Exception("对不起,指定范围的随机数全部产生过了。")
        
else
          
{
            Random r 
= new Random();
           
int index=(int)(r.NextDouble()*10+1);
            
int returnnum =(int)(nums[index]);
            nums.RemoveAt(index);
            return returnnum;
            }

    }


}

你可能感兴趣的:(随机数)