在vb.net中洗牌一串字符串

Abd*_*mad 3 .net vb.net arrays

我正在vb.net中开发一个网页,它将为用户生成一些多项选择题.我需要将已经放入数组的四个答案洗牌.让我们假设我必须遵循数组:

array = {"Correct", "Wrong1", "Wrong2", "Wrong3"}
Run Code Online (Sandbox Code Playgroud)

我试着使用以下方法:

Public Shared Function Shuffle(ByVal items() As String) As Array
        Dim max_index As Integer = items.Length - 1
        Dim rnd As New Random(DateTime.Now.Millisecond)
        For i As Integer = 0 To max_index
            ' Pick an item for position i.
            Randomize()
            Dim j As Integer = rnd.Next(i, max_index)
            ' Swap them.
            Dim temp As String = items(i)
            items(i) = items(j)
            items(j) = temp
        Next i
        Return items
    End Function
Run Code Online (Sandbox Code Playgroud)

该功能工作得非常好,但我的问题是,如果我有四个问题,每个问题的答案都会被洗牌,但正确的答案将在一个位置,如:

    array = {"Wrong1", "Correct", "Wrong2", "Wrong3"}
    array = {"Wrong2", "Correct", "Wrong3", "Wrong1"}
    array = {"Wrong3", "Correct", "Wrong1", "Wrong2"}
    array = {"Wrong1", "Correct", "Wrong3", "Wrong2"}
Run Code Online (Sandbox Code Playgroud)

我需要的是将其位置从一个问题改为另一个问题的正确答案.非常感谢您的帮助.

Ňɏs*_*arp 6

您的Shuffle方法和使用有几个问题Random.

  1. Randomize用于旧的旧版VB Rnd功能.它对闪亮的新.NET Random类没有影响.
  2. 很少需要提供定制种子; 事实上,这可能对你不利.
  3. 创建一个Random供整个应用程序使用而不是每次随机播放(或点击某些内容).永远不要在循环中创建它们 - 这几乎可以保证重复的数字.

  4. 最大参数Random.Next(min, max)独占的,因此您的范围实际上比它应该小1个元素.

  5. 你的Shuffle非常接近,但有一些缺陷:
    • 一般来说,Fisher-Yates Shuffle的NET版本是一个shuffle(Sub)使它非常有效(而不是返回一个新的集合)
    • 通过列表或数组向后循环非常重要.1

标准Fisher-Yates Shuffle:

' form/class level var
Private rnd As New Random()

Public Sub Shuffle(items As String())
    Dim j As Int32
    Dim temp As String

    For n As Int32 = items.Length - 1 To 0 Step -1
        j = rnd.Next(0, n + 1)
        ' Swap them.
        temp = items(n)
        items(n) = items(j)
        items(j) = temp
    Next n
End Sub
Run Code Online (Sandbox Code Playgroud)

从同一起始阵列的4次shuffle输出:

Shuffle #1    Wrong3   Correct  Wrong2   Wrong1   
Shuffle #2    Correct  Wrong2   Wrong1    Wrong3   
Shuffle #3    Wrong2   Correct  Wrong3    Wrong1   
Shuffle #4    Correct  Wrong1   Wrong2    Wrong3   
Shuffle #5    Correct  Wrong1   Wrong3    Wrong2   
Run Code Online (Sandbox Code Playgroud)

变化

Random您可以传递它而不是全局生成器(作为扩展方法有用):

Public Sub Shuffle(items As String(), RNG As Random)
Run Code Online (Sandbox Code Playgroud)

改组多种类型的通用方法:

' Generic shuffle for basic type arrays
Public Sub Shuffle(Of T)(items As T(), rng As Random)
    Dim temp As T
    Dim j As Int32

    For i As Int32 = items.Count - 1 To 0 Step -1
        ' Pick an item for position i.
        j = rng.Next(i + 1)
        ' Swap 
        temp = items(i)
        items(i) = items(j)
        items(j) = temp
    Next i
End Sub
Run Code Online (Sandbox Code Playgroud)

例子:

Shuffle(intArray, myRnd)
Shuffle(strArray, myRnd)
Shuffle(colors, myRnd)
Shuffle(myButtons, myRnd)
Run Code Online (Sandbox Code Playgroud)

简单的重新排序

最后,对于一些您可能只重新排序一次的简单内容,使用扩展方法的简单且通常足够好的版本:

Dim ShuffledItems = myItems.OrderBy(Function() rnd.Next).ToArray()
Run Code Online (Sandbox Code Playgroud)

这样代码较少,易于删除,但效率得多.

1您想要向后循环的原因是将每个数组元素限制为一个交换.一旦"X"被移动到items(j)在最后一步,该位置是从考虑中删除,因为rnd.Next(0, n + 1)将仅挑最后一个元素/循环索引.许多实现都错了.

  • 很好的解释,很好的例子和一个很棒的人!谢谢! (2认同)