SUMIFS using Dictionary

inactiveUser462638

New Member
Joined
Jun 13, 2020
Messages
13
Office Version
  1. 2016
Platform
  1. Windows
Hi Excel Champs,

Please help me getting this calculated. I'm using SUMIFS formulas now but it's going very slow.

I have data like this:

Element TypeInflow/OutflowForecast IDElement DescriptionDateRecording PersonVendor / CustomerGross Amount
HR CostsOutflowRefund Andre01.01.2020PhilCustomer118.00
Technology ServicesOutflowSubscription01.01.2020PhilCustomer245.00
Loans and TaxesInflowSubscription01.02.2020PhilCustomer245.00
Office CostsOutflowSubscription01.03.2020PhilCustomer236.00
Travel, Accommodation, ExpensesOutflowIncome Tax01.04.2020PhilCustomer31337.79
Technology ServicesInflowSubscription01.04.2020PhilCustomer236.00
Loans and TaxesOutflowTo Customer401.05.2020PhilCustomer40.00
Loans and TaxesInflowSubscription01.05.2020PhilCustomer236.00

I need to make a CashFlow report out of above data like this:


DateBeginning BalanceCash InflowsCash OutflowsNet Cash FlowsEnding Cash Balance
01.01.2020100000=SUMIF()=SUMIF()=C2-D2=B2+E2
02.01.2020=F2=SUMIF()=SUMIF()=C3-D3=B3+E3
03.01.2020=F3=SUMIF()=SUMIF()=C4-D4=B4+E4

Can you please help me with a VBA code that calculates this using the Dictionary?

Thank you very much!
 
Last edited by a moderator:

Excel Facts

Bring active cell back into view
Start at A1 and select to A9999 while writing a formula, you can't see A1 anymore. Press Ctrl+Backspace to bring active cell into view.
Agree. Then try the following.
Only in cell B2 of the "Calculation Sheet Final Form" put the Beginning Balance.

The macro will work according to the structure of the sheets shown in post #9

VBA Code:
Sub SUMIFS_using_Dictionary()
  Dim a As Variant, b As Variant, ky As Variant
  Dim dic As Object, col As Object
  Dim i As Long, j As Long, k As Long, lr As Long, lc As Long
  Dim m As Double, n As Double, bb As Double
  
  Set dic = CreateObject("Scripting.Dictionary")
  Set col = CreateObject("System.Collections.ArrayList")
  bb = Sheets("Calculation Sheet Final Form").Range("B2").Value 'Beginning Balance
  
  a = Sheets("Forecast Elements").Range("A2:N" & Sheets("Forecast Elements").Range("A" & Rows.Count).End(3).Row).Value2
  ReDim b(1 To UBound(a), 1 To 6)
  
  For i = 1 To UBound(a, 1)
    dic(a(i, 5)) = Empty
  Next
  '
  'sort by dates
  For Each ky In dic.keys
    col.Add ky
  Next
  col.Sort
  dic.RemoveAll
  For Each ky In col
    dic(ky) = "0|0"
  Next ky
  'end sort
  '
  For i = 1 To UBound(a, 1)
    m = Split(dic(a(i, 5)), "|")(0)
    n = Split(dic(a(i, 5)), "|")(1)
    If LCase(a(i, 1)) = LCase("Inflow") Then m = m + a(i, 8) Else n = n + a(i, 8)
    dic(a(i, 5)) = m & "|" & n
  Next
  '
  For Each ky In dic.keys
    j = j + 1
    b(j, 1) = ky
    b(j, 2) = bb
    b(j, 3) = Split(dic(ky), "|")(0)
    b(j, 4) = Split(dic(ky), "|")(1)
    b(j, 5) = b(j, 3) - b(j, 4)
    b(j, 6) = b(j, 2) + b(j, 5)
    bb = b(j, 6)
  Next
  '
  Sheets("Calculation Sheet Final Form").Range("A2").Resize(dic.Count, 6).Value = b
End Sub
 
Upvote 0
Hello @DanteAmor!

Can you please put some comments on the code when you have time? I tried to understand it but I'm not doing a great job. I could not find what that things do: dic(ky) = "0|0", Split(dic(a(i, 5)), "|")(0).

Thank you very much!
 
Upvote 0
I hope the following helps you.

VBA Code:
Sub SUMIFS_using_Dictionary()
  Dim a As Variant, b As Variant, ky As Variant
  Dim dic As Object, col As Object
  Dim i As Long, j As Long, k As Long, lr As Long, lc As Long
  Dim m As Double, n As Double, bb As Double
  
  Set dic = CreateObject("Scripting.Dictionary")
  Set col = CreateObject("System.Collections.ArrayList")
  bb = Sheets("Calculation Sheet Final Form").Range("B2").Value 'Beginning Balance
  
  a = Sheets("Forecast Elements").Range("A2:N" & Sheets("Forecast Elements").Range("A" & Rows.Count).End(3).Row).Value2
  ReDim b(1 To UBound(a), 1 To 6)
  
  'Fill the dictionary, only the unique dates are loaded.
  For i = 1 To UBound(a, 1)
    'dic(key)  = item 'In this case it is not necessary to fill an item, that's why I put empty
    dic(a(i, 5)) = Empty
  Next
  'In the next part the dates are sorted.
  'sort by dates
  For Each ky In dic.keys
    col.Add ky
  Next
  col.Sort
  'clean dictioary
  dic.RemoveAll
  'Fill the dictionary, With sorted dates
  For Each ky In col
    'key    = item
    'On each date I add a zero for the sum of the inflow
    'and another zero for the sum of the outflow
    'It is like initializing sums.
    dic(ky) = "0|0"
  Next ky
  'end sort
  '
  'Cycle to go through all the dates
  For i = 1 To UBound(a, 1)
    'I get the accumulated value of the sums from the dictionary.
    'Check the value of column 1, if it is "Inflow"
    'Assuming that in column 1 you have "Inflow",
    '              in column 5 you have 23/Jun/2020 and
    '              in column 8 you have the value 100
    '(Obviously the first time it returns 0 and 0)
    m = Split(dic(a(i, 5)), "|")(0)   'get the cumulative value of the inflow sum
    'm = split(dic(23/Jun/2020)), "|")(0)
    'm = split("0|0")), "|")(0) 'The (0) is to obtain the first value of the split
    'm = 0
    n = Split(dic(a(i, 5)), "|")(1)   'The (1) is to obtain the second value of the split
    'Then m = 0 and n = 0
    '
    If LCase(a(i, 1)) = LCase("Inflow") Then
      'then accumulate the value of column 8 in variable m
      m = m + a(i, 8)
      'm = 0 + 100
      'm = 100
    Else
      'If not, accumulate in variable n
      n = n + a(i, 8)
    End If
    'Return accumulated values to dictionary
    dic(a(i, 5)) = m & "|" & n
    'dic(23/Jun/2020) = 100 & "|" & 0
    'dic(23/Jun/2020) = "100|0"
        
    '**********
    'In the next cycle if there is another date in the data with Jun/23
    'then m = 100 and n = 0
    'if the data is "inflow" let's say 345, then m = 100 + 345
    'Return accumulated values to dictionary
    'dic(23/Jun/2020) = "345|0"
  Next
  '
  'Cycle to fill array b by scrolling through values sums stored in dictionary
  For Each ky In dic.keys
    j = j + 1 'row counter
    b(j, 1) = ky  'the date
    b(j, 2) = bb  'Beginning Balance value.
    b(j, 3) = Split(dic(ky), "|")(0)  'cumulative value for inflow
    b(j, 4) = Split(dic(ky), "|")(1)  'cumulative value for outflow
    b(j, 5) = b(j, 3) - b(j, 4) 'inflow - outflow
    b(j, 6) = b(j, 2) + b(j, 5) 'Beginning Balance + difference (inflow - outflow)
    bb = b(j, 6)  'update the value of bb with the sum (Beginning Balance + difference (inflow - outflow))
  Next
  '
  'output array  b
  Sheets("Calculation Sheet Final Form").Range("A2").Resize(dic.Count, 6).Value = b
End Sub
 
Upvote 0

Forum statistics

Threads
1,224,827
Messages
6,181,195
Members
453,021
Latest member
pingpong7117

We've detected that you are using an adblocker.

We have a great community of people providing Excel help here, but the hosting costs are enormous. You can help keep this site running by allowing ads on MrExcel.com.
Allow Ads at MrExcel

Which adblocker are you using?

Disable AdBlock

Follow these easy steps to disable AdBlock

1)Click on the icon in the browser’s toolbar.
2)Click on the icon in the browser’s toolbar.
2)Click on the "Pause on this site" option.
Go back

Disable AdBlock Plus

Follow these easy steps to disable AdBlock Plus

1)Click on the icon in the browser’s toolbar.
2)Click on the toggle to disable it for "mrexcel.com".
Go back

Disable uBlock Origin

Follow these easy steps to disable uBlock Origin

1)Click on the icon in the browser’s toolbar.
2)Click on the "Power" button.
3)Click on the "Refresh" button.
Go back

Disable uBlock

Follow these easy steps to disable uBlock

1)Click on the icon in the browser’s toolbar.
2)Click on the "Power" button.
3)Click on the "Refresh" button.
Go back
Back
Top