.net - 在 vb.net 中对字符串数组进行洗牌

标签 .net arrays vb.net

我正在 vb.net 中开发一个网页,该网页将为用户生成许多多项选择题。我需要对已经放入数组中的四个答案进行洗牌。让我们假设我必须关注数组:

array = {"Correct", "Wrong1", "Wrong2", "Wrong3"}

我尝试使用以下方法:

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

这个功能工作得很好,但我的问题是,如果我有四个问题,每个问题的答案都会被打乱,但正确的答案会在一个位置,比如:

    array = {"Wrong1", "Correct", "Wrong2", "Wrong3"}
    array = {"Wrong2", "Correct", "Wrong3", "Wrong1"}
    array = {"Wrong3", "Correct", "Wrong1", "Wrong2"}
    array = {"Wrong1", "Correct", "Wrong3", "Wrong2"}

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

最佳答案

您的 Shuffle 方法和 Random 的使用存在几个问题。

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

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

  5. 您的 Shuffle 非常接近,但有一些缺陷:
    • 一般是 NET 版本的 Fisher-Yates Shuffle是一个就地洗牌 (Sub) 使其非常高效(而不是返回一个新集合)
    • 通过列表或数组循环向后很重要。1

标准的 Fisher-Yates 洗牌:

' 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

相同起始数组的 4 次随机播放的输出:

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   

变化

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

Public Sub Shuffle(items As String(), RNG As Random)

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

' 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

例子:

Shuffle(intArray, myRnd)
Shuffle(strArray, myRnd)
Shuffle(colors, myRnd)
Shuffle(myButtons, myRnd)

简单的重新排序

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

Dim ShuffledItems = myItems.OrderBy(Function() rnd.Next).ToArray()

这有更少的代码,很容易冲刺,但很多效率低下。

1 你想向后循环的原因是限制每个数组元素一个交换。一旦在最后一步将“X”移动到 items(j),该位置将不再考虑,因为 rnd.Next(0, n + 1) 只会选择直到最后一个元素/循环索引。许多实现都犯了这个错误。

关于.net - 在 vb.net 中对字符串数组进行洗牌,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/29358857/

相关文章:

c# - 如何取消发布 .Net 应用程序

.NET 版本的 Java 断言 : Unit tesing memory leaks

javascript - 如何在更改或选择事件时禁用多选?

java - 数组边界检查可以静态执行吗?

.net - MSBuild 编译错误怪异

.net - .NET 中究竟在什么情况下会抛出 SecurityException?

java - Libgdx 数组 |删除全部导致错误

xml - 如何确定 XmlNode 是否具有特定属性?

javascript - 下载在 GridView 中选中的文件的更好方法

c# - 什么时候创建变量?