繁体   English   中英

在vb.net中对类数组进行排序

[英]To sort a class array in vb.net

我花了两天的时间在互联网上搜索,试图找到解决方案来简单地排序一个由一类字符串和整数组成的数组(只有一个可能包含不规则字符的字符串元素)。 请帮忙! 我已经基于microsoft示例创建了一个简化的代码,我想要做的是:

Public Class Form1
    Class car
        Public Make As String = ""
        Public Year As Integer = 0
    End Class

    Private Sub Button1_Click(sender As Object, e As EventArgs) Handles Button1.Click
        Dim arrayOfCars() As car
        Dim arrayElement As Integer = 0

        'Exploded simplified loop to fill the array (original has 20 objects in the class
        'and the array grows depending on input to no more than a few hundred.
        ReDim arrayOfCars(0)
        arrayOfCars(0) = New car
        arrayOfCars(0).Make = "Ford"
        arrayOfCars(0).Year = 1992
        ReDim Preserve arrayOfCars(1)
        arrayOfCars(1) = New car
        arrayOfCars(1).Make = "Fiat"
        arrayOfCars(1).Year = 1988
        ReDim Preserve arrayOfCars(2)
        arrayOfCars(2) = New car
        arrayOfCars(2).Make = "Buick"
        arrayOfCars(2).Year = 1932
        ReDim Preserve arrayOfCars(3)
        arrayOfCars(3) = New car
        arrayOfCars(3).Make = "Ford"
        arrayOfCars(3).Year = 1932
        ReDim Preserve arrayOfCars(4)
        arrayOfCars(4) = New car
        arrayOfCars(4).Make = "Dodge"
        arrayOfCars(4).Year = 1999
        ReDim Preserve arrayOfCars(5)
        arrayOfCars(5) = New car
        arrayOfCars(5).Make = "Honda"
        arrayOfCars(5).Year = 1977

        'view array before sort
        For i = 0 To 5
            Debug.WriteLine(arrayOfCars(i).Make & vbTab & arrayOfCars(i).Year)
        Next
        Debug.WriteLine("*************************")

        'sort array by the string component [Make]
        'Array.Sort(arrayOfCars)
        'arrayOfCars = arrayOfCars.OrderBy(Function(car) car.Make)
        '????????????????

        'view array after sort
        For i = 0 To 5
            Debug.WriteLine(arrayOfCars(i).Make & vbTab & arrayOfCars(i).Year)
        Next
    End Sub
End Class

如果你想按Make属性排序,一个开箱即用的方法是使用Array.SortComparison(Of T)重载:

Array.Sort(arrayOfCars, Function(car1 As Car, car2 as Car)
                            Return car1.Make.CompareTo(car2.Make) 
                        End Function)

请注意,您应该采取哪些车是照顾Nothing或(更可能) Make -值这是Nothing 两者都会导致NullReferenceException 因此你可以使用:

Array.Sort(arrayOfCars, Function(car1 As Car, car2 As Car)
        If Object.ReferenceEquals(car1,car2)
            return 0
         Else if car1 is nothing 
            Return -1
         Else if car2 is nothing 
            Return 1
         Else
            return String.Compare(car1.Make, car2.Make)
        End If
    End Function)

LINQ是另一种(因为需要重新创建数组而效率稍低)方法:

Dim orderedCars = from car in arrayOfCars
                  order by car.Make Ascending
arrayOfCars = arrayOfCars.ToArray()

LINQ方法更易于维护且更易于阅读,但需要创建新阵列。 因此,如果您不想修改原始数组,则应使用该数组。

通常,如果要添加对象,则不应使用数组,因为数组是固定大小的集合。 使用List(Of Car)代替它,它有一个Add方法

另一个挑剔,遵循.NET命名/ 大写惯例 ,使用Car而不是car

Linq是你的朋友和OrderBy方法。 但是为了使用你的数组需要成为List

您可以轻松地将其转换为List并按照这样的make对其进行排序(不是这不会对原始列表进行排序,而是创建按您指定的条件排序的新列表):

Dim sortedListOfCars = arrayOfCars.ToList.OrderBy(Function(x) x.Make)

但是将它声明为开头的列表会更容易:

Dim listOfCars As New List(Of Car)

然后像这样添加你的车

listOfCars.Add(New Car With {.Make = "Ford", .Year = 1992})
listOfCars.Add(New Car With {.Make = "Fiat", .Year = 1988})
'etc.

然后你直接订购OrderBy:

Dim sortedListOfCars = arrayOfCars.OrderBy(Function(x) x.Make)

您可以为Array.Sort()函数实现IComparer。

Public Class CarComparer : Implements IComparer
   Function Compare(x As car, y As car) As Integer _
            Implements IComparer.Compare
      Return New CaseInsensitiveComparer().Compare(x.Make, y.Make)
   End Function 
End Class

像这样使用它:

Array.Sort(arrayOfCars, new CarComparer())

如果您经常要满足此要求,则应实现KeyComparer类。 这将允许您提出简洁的语法,例如:

Array.Sort(arrayOfCars, New KeyComparer(Function(c As car) c.Make))

Array.Sort(..., Comparison)方法的缺点是它不处理空值,并且如果任何列表元素碰巧为null,则会抛出NullReferenceException 这可能与您的案件有关,也可能与您的案件无关。

在线有很多KeyComparer实现。 我在KeyComparer文章中详细讨论了这个具体问题 - 并提供了一个示例C#实现。 VB.NET转换如下:

Public Class KeyComparer(Of TSource, TKey)
    Inherits Comparer(Of TSource)

    Private ReadOnly _keySelector As Func(Of TSource, TKey)
    Private ReadOnly _innerComparer As IComparer(Of TKey)

    Public Sub New(keySelector As Func(Of TSource, TKey), Optional innerComparer As IComparer(Of TKey) = Nothing)
        _keySelector = keySelector
        _innerComparer = If(innerComparer, Comparer(Of TKey).[Default])
    End Sub

    Public Overrides Function Compare(x As TSource, y As TSource) As Integer
        If Object.ReferenceEquals(x, y) Then
            Return 0
        End If
        If x Is Nothing Then
            Return -1
        End If
        If y Is Nothing Then
            Return 1
        End If

        Dim xKey As TKey = _keySelector(x)
        Dim yKey As TKey = _keySelector(y)
        Return _innerComparer.Compare(xKey, yKey)
    End Function
End Class

暂无
暂无

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM