VB.Net 2D Dictionary - very slow

I need to create a 2D dictionary / key pair. I've tried something like this.

enter image description here

Dim TwoDimData As New Dictionary(Of String, Dictionary(Of String, String))

'Create an empty table
For Each aid In AIDList   '(contains 15000 elements)
    TwoDimData.Add(aid, New Dictionary(Of String, String))
    For Each bid In BIDList   'contains 30 elements
        TwoDimData.Item(aid).Add(bid, "")
    Next
Next

'Later populate values.
[some code here to populate the table]

'Now access the value
'The idea is to access the info as given below (access by row name & col name)
 Msgbox TwoDimData.Item("A004").Item("B005")   ' should give the value of 2
 Msgbox TwoDimData.Item("A008").Item("B002")   ' should return empty string. No error

      

Question:

The problem is creating an empty table. It takes 70 seconds to create a TwoDimData table with empty values. Everything else seems fine. Is there a way to improve performance - maybe instead of using a dictionary?

+3


source to share


3 answers


I suggest you try it Dictionary(Of Tuple(Of String, String), String)

. That is, keys are string pairs ( Tuple(Of String, String)

) and values ​​are strings. This seems to fit well with the diagram in your question.

Dim matrix As New Dictionary(Of Tuple(Of String, String), String)

' Add a value to the matrix:
matrix.Add(Tuple.Create("A003", "B004"), "3")

' Retrieve a value from the matrix:
Dim valueAtA003B004 = matrix(Tuple.Create("A003", "B004"))

      

Of course, you can define your own key type (representing a combination of two strings) if Tuple(Of String, String)

it seems too general for your liking.

Alternatively you can also just use (possibly jagged ) 2D arrays, but that can potentially waste a lot of space if your data is sparse (i.e. if there are a lot of empty cells in that 2D matrix); and you will have to use numeric indices instead of strings.



PS: Actually, consider changing the type of a dictionary word from String

to Integer

; your example matrix assumes it only contains integers, so it might not make sense to store them as strings.

PPS: Do not add values ​​for "blank" cells to the dictionary. It would be very wasteful. Instead, instead of simply extracting a value from the dictionary, you check if the dictionary contains a key:

Dim valueA As String = ""  ' the default value
If matrix.TryGetValue(Tuple.Create("A007", "B002"), valueA) Then
    ' the given key was present, and the associated value has been retrieved
End If

      

+3


source


I would think a simple structure is enough for this?

Public Structure My2DItem
  Public Row As Integer
  Public Col As Integer
  Public Value As String
End Structure

Public My2DArray As Generic.List(Of My2DItem) = Nothing
Public Size As Integer
Public MaxRows As Integer
Public MaxCols As Integer
'
Sub Initialise2DArray()
'
Dim CountX As Integer
Dim CountY As Integer
Dim Item As My2DItem
'
  'initialise
  MaxRows = 15000
  MaxCols = 30
  Size = MaxRows * MaxCols
  My2DArray = New Generic.List(Of My2DItem)
  '   
  'Create an empty table
  For CountY = 1 To 15000
    For CountX = 1 To 30
      Item = New My2DItem
      Item.Row = CountY
      Item.Col = CountX
      Item.Value = "0"
      My2DArray.Add(Item)
      Item = Nothing
    Next
  Next
'
End Sub

      

And to read data from an array,



Function GetValue(Y As Integer, X As Integer) As String
'
Dim counter As Integer
'
GetValue = "Error!"
If My2DArray.Count > 0 Then
  For counter = 0 To My2DArray.Count - 1
    If My2DArray(counter).Row = Y Then
      If My2DArray(counter).Col = X Then
        GetValue = My2DArray(counter).Value
        Exit Function
      End If
    End If
  Next
End If
'
End Function

      

And to read your sample cell A004 B005

MyStringValue = GetValue(4,5)

      

+1


source


I would suggest creating a class that has properties for AID and BID and use that as the basis for the values ​​you want to store

Public Class AIdBId
    Public Property AId As Integer
    Public Property BId As Integer
    Public Sub New(aId As Integer, bId As Integer)
        Me.AId = aid
        Me.BId = bid
    End Sub
End Class

      

Note that I have used integers for everything because it seems like that is all you need and it is more efficient to use a string

Then you can add values ​​where they are nonzero:

'define your dictionary
Dim valueMatrix As New Dictionary(Of AIdBId, Integer)

'add your values
valueMatrix.Add(New AIdBId(1, 1), 1)
valueMatrix.Add(New AIdBId(2, 3), 1)
valueMatrix.Add(New AIdBId(4, 3), 3)
valueMatrix.Add(New AIdBId(5, 8), 8)

'check if a value exixts
Dim valueExixsts As Boolean = valueMatrix.ContainsKey(New AIdBId(9, 9))

'get a value
Dim i As Integer = valueMatrix(New AIdBId(4, 3))

      

So now you can combine the two to return a value if there is one or zero if not:

 Private Function GetValue(valuematrix As Dictionary(Of AIdBId, Integer), aId As Integer, bId As Integer) As Integer
    Dim xRef As New AIdBId(aId, bId)
    If valuematrix.ContainsKey(xRef) Then
        Return valuematrix(xRef)
    Else
        Return 0
    End If
End Function

      

0


source







All Articles