Primeira estratégia: otimizar a função em si
Deve dobrar a velocidade
Public Function Yield(Name As String, Price As Double)
Dim Lookup As Range, rw As Integer
Set Lookup = Range("LookupRange")
rw = Application.WorksheetFunction.Match(Name, Lookup.Resize(ColumnSize:=1), 0)
Yield = 100 * Application.Run("otherCustomFunction", Lookup.Cells(rw, 3), Lookup.Cells(rw, 7), Price)
End Function
Isso porque você só pesquisa o intervalo com o nome "LookupRange" uma vez em vez de duas vezes e procura apenas a linha certa uma vez em vez de duas vezes.
Segunda estratégia: recuperar o intervalo apenas uma vez adiantado
Provavelmente 4 vezes mais rápido
Se recuperarmos o intervalo no código que usa a função yield
, só precisamos fazer isso uma vez
Public Function Yield(Lookup As Range, Name As String, Price As Double)
rw = Application.WorksheetFunction.Match(Name, Lookup.Resize(ColumnSize:=1), 0)
Yield = 100 * Application.Run("otherCustomFunction", Lookup.Cells(rw, 3), Lookup.Cells(rw, 7), Price)
End Function
Public Sub CallingRoutine()
Dim Lookup As Range, rw As Integer
Set Lookup = Range("LookupRange")
' Some code
For Each someItem In someSet
Dim amount As Double, Name As String, Price As Double
' Some code to deter;ine name and price
amount = Yield(Lookup, Name, Price)
' Some code that used the yield
Next someThing
End Sub
Há uma variante dessa estratégia na qual você declara Lookup fora de todas as rotinas, como eu faço com o dicionário abaixo.
Terceira estratégia: coloque todos os valores relevantes em um dicionário
Uma ordem de grandeza mais rápida se você chamar Yield
MUITO vezes.
- Você consulta o intervalo nomeado
- Você solicita todos os valores do excel de uma só vez
- Você procura o
Name
s em um dicionário, que é muito mais eficiente do que procurar em um intervalo
Este é o código:
Public Function Yield(Name As String, Price As Double)
If LookDict Is Nothing Then
Set LookDict = New Dictionary
Dim LookVal As Variant, rw As Integer, ToUse As ToUseType
LookVal = Range("LookupRange").Value
For rw = LBound(LookVal, 1) To UBound(LookVal, 1)
Set ToUse = New ToUseType
ToUse.Row3Val = LookVal(rw, 3)
ToUse.Row7Val = LookVal(rw, 7)
LookDict.Add LookVal(rw, 1), ToUse
Next rw
End If
Set ToUse = LookDict.Item(Name)
Yield = 100 * Application.Run("otherCustomFunction", _
ToUse.Row3Val, ToUse.Row7Val, Price)
End Function
Public Sub CallingRoutine()
' Some code
For Each someItem In someSet
Dim amount As Double, Name As String, Price As Double
' Some code to deter;ine name and price
amount = Yield(Name, Price)
' Some code that used the yield
Next someThing
End Sub