System.OutOfMemoryException 和使用 StringBuilder 的并发症
System.OutOfMemoryException and complication with Using StringBuilder
下面的代码给了我System.OutOfMemoryException,我试图通过在我的代码中使用 StringBuilder 来改变它,但我无法成功使用它。
任何人都可以告诉我在我的代码中使用 StringBuilder 而不是字符串之间的许多连接的最佳方法是什么?
Dim sReport As String = ""
For i As Integer = 1 To oPuntInterpretar.Dies.Count
' DGARCIA: Per MeteoPlay3 no ens interessa que hi hagi diferència en l'hora local UTC,
' ja que cada previsió s'ofereix des del mateir punt evaluat.
oPuntInterpretar.DifHoraLocalUTC = 0
If (oConfiguracio.Opcions_Meteo3D.LogPGs = True) Then
' Dades del report...
sReport = "ID_PUNTGEOGRAFIC;" & oPuntInterpretar.IPuntGeografic & vbCrLf & _
"LATITUD_PUNTGEOGRAFIC;" & Math.Round(oPuntInterpretar.Latitud, 2) & vbCrLf & _
"LONGITUD_PUNTGEOGRAFIC;" & Math.Round(oPuntInterpretar.Longitud, 2) & vbCrLf & _
"ALCADA_PUNTGEOGRAFIC (metres);" & oPuntInterpretar.Alçada & vbCrLf & _
"TIPUS_PUNTGEOGRAFIC;" & IIf(oPuntInterpretar.Tipus = clsPuntGeografic.TipusPunt.Poblacio, "POBLACIO", "MUNTANYA") & vbCrLf & _
"DIF_HORALOCAL_UTC;" & oPuntInterpretar.DifHoraLocalUTC & vbCrLf & _
"DATA_MODEL;" & dtDataElaboracio.ToString & vbCrLf & _
"DATA_VALIDESA;" & oPuntInterpretar.Dies(i).DataValidesa.ToString & vbCrLf & vbCrLf
End If
' a + a + de les hores calculem TMax i TMin
oPuntInterpretar.CalculaTempMin(oConfiguracio, oPuntInterpretar.Dies(i).FranjaMati, oPuntInterpretar.VariablesPG, oPuntInterpretar.DifHoraLocalUTC, oPuntInterpretar.Alçada, i - 1, iHoresRestar, sReport)
sReport &= vbCrLf
oPuntInterpretar.CalculaTempMax(oConfiguracio, oPuntInterpretar.Dies(i).FranjaMati, oPuntInterpretar.VariablesPG, oPuntInterpretar.DifHoraLocalUTC, oPuntInterpretar.Alçada, i - 1, iHoresRestar, sReport)
sReport &= vbCrLf
' Ara mirem si tenim hores per aquest dia...
For j As Integer = 1 To oPuntInterpretar.Dies(i).Hores.count
With oPuntInterpretar
' VentVel (km/hora -> passar a Beaufort!)
' VentDir (ºC -> passar a CodiVent)
' Després caldrà juntar els dos valors a CodiVent per insertar en base de dades!
.CalculaVentDir(oConfiguracio, .Dies(i).Hores(j), .VariablesPG, .DifHoraLocalUTC, .Alçada, i - 1, iHoresRestar, sReport, False)
sReport &= vbCrLf
.CalculaVentVel(oConfiguracio, .Dies(i).Hores(j), .VariablesPG, .DifHoraLocalUTC, .Alçada, i - 1, iHoresRestar, sReport, False)
sReport &= vbCrLf
If (.Dies(i).Hores(j).VentVel <> clsConfiguracio.NO_DADA_DOUBLE) And (.Dies(i).Hores(j).VentDir <> clsConfiguracio.NO_DADA_DOUBLE) Then
.Dies(i).Hores(j).CodiVent = CalculaDirTaula(oConfiguracio, .Dies(i).Hores(j).VentDir) & GrauBeaufort(.Dies(i).Hores(j).VentVel, oConfiguracio.TaulaGrauBeaufort)
sReport &= vbCrLf
Else
.Dies(i).Hores(j).CodiVent = ""
End If
.CalculaTempSuperficie(oConfiguracio, .Dies(i).Hores(j), .VariablesPG, .DifHoraLocalUTC, .Alçada, i - 1, iHoresRestar, sReport, False)
sReport &= vbCrLf
If (j = 1) Then
.CalculaPluja(oConfiguracio, .Dies(i).Hores(j), .Dies(i).Hores(j).tipusHora, .VariablesPG, .DifHoraLocalUTC, .Alçada, i - 1, iHoresRestar, sReport, False)
sReport &= vbCrLf
Else
.CalculaPluja(oConfiguracio, .Dies(i).Hores(j), .Dies(i).Hores(j - 1).tipusHora + 1, .VariablesPG, .DifHoraLocalUTC, .Alçada, i - 1, iHoresRestar, sReport, False)
sReport &= vbCrLf
End If
.CalculaPressio(oConfiguracio, .Dies(i).Hores(j), .VariablesPG, .DifHoraLocalUTC, .Alçada, i - 1, iHoresRestar, sReport, False)
sReport &= vbCrLf
.CalculaNuvolositat(oConfiguracio, .Dies(i).Hores(j), .VariablesPG, .DifHoraLocalUTC, .Alçada, i - 1, iHoresRestar, sReport, False)
sReport &= vbCrLf
.CalculaSimbol(oConfiguracio, .Dies(i).Hores(j), .VariablesPG, .DifHoraLocalUTC, .Alçada, i - 1, moDB, iHoresRestar, sReport, False, True)
sReport &= vbCrLf
End With
我试过这样使用它,但它似乎不起作用。我猜是关于 &=
运算符的问题,或者是附加错误。
修改后的代码:
Dim sReport As new StringBuilder
Try
iHoresRestar = dtDataElaboracio.Hour
' Per cada dia, calculem els valors de les tres franges i totes les hores...
For i As Integer = 1 To oPuntInterpretar.Dies.Count
' DGARCIA: Per MeteoPlay3 no ens interessa que hi hagi diferència en l'hora local UTC,
' ja que cada previsió s'ofereix des del mateir punt evaluat.
oPuntInterpretar.DifHoraLocalUTC = 0
If (oConfiguracio.Opcions_Meteo3D.LogPGs = True) Then
' Dades del report...
sReport.Append("ID_PUNTGEOGRAFIC;" & oPuntInterpretar.IPuntGeografic & vbCrLf & _
"LATITUD_PUNTGEOGRAFIC;" & Math.Round(oPuntInterpretar.Latitud, 2) & vbCrLf & _
"LONGITUD_PUNTGEOGRAFIC;" & Math.Round(oPuntInterpretar.Longitud, 2) & vbCrLf & _
"ALCADA_PUNTGEOGRAFIC (metres);" & oPuntInterpretar.Alçada & vbCrLf & _
"TIPUS_PUNTGEOGRAFIC;" & IIf(oPuntInterpretar.Tipus = clsPuntGeografic.TipusPunt.Poblacio, "POBLACIO", "MUNTANYA") & vbCrLf & _
"DIF_HORALOCAL_UTC;" & oPuntInterpretar.DifHoraLocalUTC & vbCrLf & _
"DATA_MODEL;" & dtDataElaboracio.ToString & vbCrLf & _
"DATA_VALIDESA;" & oPuntInterpretar.Dies(i).DataValidesa.ToString & vbCrLf & vbCrLf)
End If
' a + a + de les hores calculem TMax i TMin
oPuntInterpretar.CalculaTempMin(oConfiguracio, oPuntInterpretar.Dies(i).FranjaMati, oPuntInterpretar.VariablesPG, oPuntInterpretar.DifHoraLocalUTC, oPuntInterpretar.Alçada, i - 1, iHoresRestar, sReport.ToString)
sReport.AppendLine()
oPuntInterpretar.CalculaTempMax(oConfiguracio, oPuntInterpretar.Dies(i).FranjaMati, oPuntInterpretar.VariablesPG, oPuntInterpretar.DifHoraLocalUTC, oPuntInterpretar.Alçada, i - 1, iHoresRestar, sReport.ToString)
sReport.AppendLine()
更新------------------------
这是将 sReport 传递给它的函数之一:
Public Function CalculaVentDir(ByVal config As clsConfiguracio, _
ByRef unaHora As clsHora, _
ByVal colVars As clsVariables, _
ByVal DifHoraLocalUTC As Integer, _
ByVal Alçada As Double, _
ByVal difDies As Integer, _
ByVal horaModel As Integer, _
ByRef strReport As String, _
Optional ByVal UtilitzemDifHoraLocalUTC As Boolean = True _
) As Boolean
Dim U, V As Double
Dim angle As Double
Dim horaIni, posU, posV As Integer
Dim ventDir As Double
Dim strMsg As String
Dim strTmp(7) As String
Try
If (colVars.CercaVariable(New clsVariable(33, 105, 10), posU) = True) And _
(colVars.CercaVariable(New clsVariable(34, 105, 10), posV) = True) _
Then
If (UtilitzemDifHoraLocalUTC = True) Then
horaIni = unaHora.tipusHora - DifHoraLocalUTC + difDies * 24 - horaModel
Else
horaIni = unaHora.tipusHora + difDies * 24 - horaModel
End If
strTmp(0) = "DIRECCIO DEL VENT"
strTmp(1) = "Hora local"
strTmp(2) = "Hora UTC"
strTmp(3) = "Component U"
strTmp(4) = "Component V"
strTmp(5) = "Angle exacte"
strTmp(6) = "Angle segons quadrant"
strTmp(7) = "Angle aproximat"
strMsg = "|Hora=" & unaHora.tipusHora & "|difDies=" & difDies & "|difHoraLocalUTC=" & DifHoraLocalUTC & "|horaModel=" & horaModel & "|horaIni=" & horaIni
strTmp(1) &= ";" & unaHora.tipusHora
strTmp(2) &= ";" & horaIni
If (horaIni < 0) Then
If (config.Opcions_Meteo3D.LogPGs = True) Then
RaiseEvent GetWarning("MESSAGE=No hem calculat la VentDir perquè la hora està fora dels marges" & strMsg & "|FUNCIO=CalculaVentDir_Hora", Me)
End If
unaHora.VentDir = clsConfiguracio.NO_DADA_DOUBLE
CalculaVentDir = False
End If
U = colVars.Valors(posU - 1, horaIni)
V = colVars.Valors(posV - 1, horaIni)
strTmp(3) &= ";" & U
strTmp(4) &= ";" & V
If (U = clsConfiguracio.NO_DADA_DOUBLE) Or (V = clsConfiguracio.NO_DADA_DOUBLE) Then
If (config.Opcions_Meteo3D.LogPGs = True) Then
RaiseEvent GetWarning("MESSAGE=No Assignem VentDir perquè no hem trobat dades de la hora " & horaIni & strMsg & "|FUNCIO=CalculaVentDir_Hora", Me)
End If
unaHora.VentDir = clsConfiguracio.NO_DADA_DOUBLE
CalculaVentDir = False
Else
angle = Math.Atan2(Math.Abs(V), Math.Abs(U)) * 180 / Math.PI
strTmp(5) &= ";" & angle
If (U = 0) Then
If (V >= 0) Then ventDir = 180
If (V < 0) Then ventDir = 0
Else
If (U > 0) And (V > 0) Then ventDir = 270 - angle
If (U < 0) And (V < 0) Then ventDir = 90 - angle
If (U > 0) And (V < 0) Then ventDir = 270 + angle
If (U < 0) And (V > 0) Then ventDir = 90 + angle
End If
strTmp(6) &= ";" & angle
unaHora.VentDir = ventDir
strTmp(7) &= ";" & angle
CalculaVentDir = True
End If
For j As Integer = 0 To UBound(strTmp)
strReport &= strTmp(j) & vbCrLf
Next
Else
If (config.Opcions_Meteo3D.LogPGs = True) Then
RaiseEvent GetWarning("MESSAGE=No Assignem VentDir perquè no hem trobat les variables 33 o 34|FUNCIO=CalculaVentDir_Hora", Me)
End If
CalculaVentDir = False
End If
Catch ex As Exception
If (config.Opcions_Meteo3D.LogPGs = True) Then
RaiseEvent GetError("MESSAGE=" & ex.Message & ", SOURCE=" & ex.StackTrace & ", FUNCIO=CalculaVentDir_Hora", Me)
End If
unaHora.VentDir = clsConfiguracio.NO_DADA_DOUBLE
CalculaVentDir = False
End Try
End Function
问题出在这些行中:
oPuntInterpretar.CalculaTempMin(oConfiguracio, oPuntInterpretar.Dies(i).FranjaMati, oPuntInterpretar.VariablesPG, oPuntInterpretar.DifHoraLocalUTC, oPuntInterpretar.Alçada, i - 1, iHoresRestar, sReport.ToString)
oPuntInterpretar.CalculaTempMax(oConfiguracio, oPuntInterpretar.Dies(i).FranjaMati, oPuntInterpretar.VariablesPG, oPuntInterpretar.DifHoraLocalUTC, oPuntInterpretar.Alçada, i - 1, iHoresRestar, sReport.ToString)
... 类似的,你将 sReport.ToString()
传递给一个方法。
您需要将这些功能更改为甚至不要求报告。如果他们需要向报告中添加一些内容,请让他们 return 一个字符串,然后在调用方法时附加结果:
sReport.Appdend(oPuntInterpretar.CalculaTempMin(oConfiguracio, oPuntInterpretar.Dies(i).FranjaMati, oPuntInterpretar.VariablesPG, oPuntInterpretar.DifHoraLocalUTC, oPuntInterpretar.Alçada, i - 1, iHoresRestar))
您也可以更改方法以改为接受 StringBuilder 对象,但这种方式是更好的编码风格。该方法并不真正需要知道报告字符串中已有的内容,函数的参数超过 3 或 4 个都是代码异味。
如果此报告很大,即使让 StringBuilder 正常工作也可能无法解决您的 OutOfMemory 问题。如果问题仍然存在,下一个要查看的地方是使用 Stream
,例如 FileStream
、ASP.Net ReponseStream
,或 Console.Out
流,所有这些都可以通过 TextWriter
.
抽象出来
最后,我们可以改进初始的 StringBuilder 翻译:
Dim reportBase As String = "ID_PUNTGEOGRAFIC;{0}{1}LATITUD_PUNTGEOGRAFIC;{2}{1}LONGITUD_PUNTGEOGRAFIC;{3}{1}ALCADA_PUNTGEOGRAFIC (metres);{4}{1}TIPUS_PUNTGEOGRAFIC;{5}{1}DIF_HORALOCAL_UTC;{6}{1}DATA_MODEL;{7}{1}DATA_VALIDESA;{8}{1}{1}"
Dim Report As New StringBuilder()
iHoresRestar = dtDataElaboracio.Hour
For Each die In oPuntInterpretar.Dies
If oConfiguracio.Opcions_Meteo3D.LogPGs Then
Report.AppendFormat(reportBase, _
oPuntInterpretar.IPuntGeografic,vbCrLf, _
Math.Round(oPuntInterpretar.Latitud, 2), _
Math.Round(oPuntInterpretar.Longitud, 2), _
oPuntInterpretar.Alçada, _
IIf(oPuntInterpretar.Tipus = clsPuntGeografic.TipusPunt.Poblacio, "POBLACIO", "MUNTANYA"), _
oPuntInterpretar.DifHoraLocalUTC, _
dtDataElaboracio.ToString, _
die.DataValidesa.ToString )
End If
'...
如果您能够更新到 Visual Studio 的更新版本(现在在大多数情况下 免费 ),我们可以进一步改进代码。我现在将特别添加 Microsoft 建议 反对 在变量名称上使用匈牙利符号疣。现在只是 report
,而不是 sReport
,puntInterpretar
而不是 oPuntInterpretar
。
下面的代码给了我System.OutOfMemoryException,我试图通过在我的代码中使用 StringBuilder 来改变它,但我无法成功使用它。
任何人都可以告诉我在我的代码中使用 StringBuilder 而不是字符串之间的许多连接的最佳方法是什么?
Dim sReport As String = ""
For i As Integer = 1 To oPuntInterpretar.Dies.Count
' DGARCIA: Per MeteoPlay3 no ens interessa que hi hagi diferència en l'hora local UTC,
' ja que cada previsió s'ofereix des del mateir punt evaluat.
oPuntInterpretar.DifHoraLocalUTC = 0
If (oConfiguracio.Opcions_Meteo3D.LogPGs = True) Then
' Dades del report...
sReport = "ID_PUNTGEOGRAFIC;" & oPuntInterpretar.IPuntGeografic & vbCrLf & _
"LATITUD_PUNTGEOGRAFIC;" & Math.Round(oPuntInterpretar.Latitud, 2) & vbCrLf & _
"LONGITUD_PUNTGEOGRAFIC;" & Math.Round(oPuntInterpretar.Longitud, 2) & vbCrLf & _
"ALCADA_PUNTGEOGRAFIC (metres);" & oPuntInterpretar.Alçada & vbCrLf & _
"TIPUS_PUNTGEOGRAFIC;" & IIf(oPuntInterpretar.Tipus = clsPuntGeografic.TipusPunt.Poblacio, "POBLACIO", "MUNTANYA") & vbCrLf & _
"DIF_HORALOCAL_UTC;" & oPuntInterpretar.DifHoraLocalUTC & vbCrLf & _
"DATA_MODEL;" & dtDataElaboracio.ToString & vbCrLf & _
"DATA_VALIDESA;" & oPuntInterpretar.Dies(i).DataValidesa.ToString & vbCrLf & vbCrLf
End If
' a + a + de les hores calculem TMax i TMin
oPuntInterpretar.CalculaTempMin(oConfiguracio, oPuntInterpretar.Dies(i).FranjaMati, oPuntInterpretar.VariablesPG, oPuntInterpretar.DifHoraLocalUTC, oPuntInterpretar.Alçada, i - 1, iHoresRestar, sReport)
sReport &= vbCrLf
oPuntInterpretar.CalculaTempMax(oConfiguracio, oPuntInterpretar.Dies(i).FranjaMati, oPuntInterpretar.VariablesPG, oPuntInterpretar.DifHoraLocalUTC, oPuntInterpretar.Alçada, i - 1, iHoresRestar, sReport)
sReport &= vbCrLf
' Ara mirem si tenim hores per aquest dia...
For j As Integer = 1 To oPuntInterpretar.Dies(i).Hores.count
With oPuntInterpretar
' VentVel (km/hora -> passar a Beaufort!)
' VentDir (ºC -> passar a CodiVent)
' Després caldrà juntar els dos valors a CodiVent per insertar en base de dades!
.CalculaVentDir(oConfiguracio, .Dies(i).Hores(j), .VariablesPG, .DifHoraLocalUTC, .Alçada, i - 1, iHoresRestar, sReport, False)
sReport &= vbCrLf
.CalculaVentVel(oConfiguracio, .Dies(i).Hores(j), .VariablesPG, .DifHoraLocalUTC, .Alçada, i - 1, iHoresRestar, sReport, False)
sReport &= vbCrLf
If (.Dies(i).Hores(j).VentVel <> clsConfiguracio.NO_DADA_DOUBLE) And (.Dies(i).Hores(j).VentDir <> clsConfiguracio.NO_DADA_DOUBLE) Then
.Dies(i).Hores(j).CodiVent = CalculaDirTaula(oConfiguracio, .Dies(i).Hores(j).VentDir) & GrauBeaufort(.Dies(i).Hores(j).VentVel, oConfiguracio.TaulaGrauBeaufort)
sReport &= vbCrLf
Else
.Dies(i).Hores(j).CodiVent = ""
End If
.CalculaTempSuperficie(oConfiguracio, .Dies(i).Hores(j), .VariablesPG, .DifHoraLocalUTC, .Alçada, i - 1, iHoresRestar, sReport, False)
sReport &= vbCrLf
If (j = 1) Then
.CalculaPluja(oConfiguracio, .Dies(i).Hores(j), .Dies(i).Hores(j).tipusHora, .VariablesPG, .DifHoraLocalUTC, .Alçada, i - 1, iHoresRestar, sReport, False)
sReport &= vbCrLf
Else
.CalculaPluja(oConfiguracio, .Dies(i).Hores(j), .Dies(i).Hores(j - 1).tipusHora + 1, .VariablesPG, .DifHoraLocalUTC, .Alçada, i - 1, iHoresRestar, sReport, False)
sReport &= vbCrLf
End If
.CalculaPressio(oConfiguracio, .Dies(i).Hores(j), .VariablesPG, .DifHoraLocalUTC, .Alçada, i - 1, iHoresRestar, sReport, False)
sReport &= vbCrLf
.CalculaNuvolositat(oConfiguracio, .Dies(i).Hores(j), .VariablesPG, .DifHoraLocalUTC, .Alçada, i - 1, iHoresRestar, sReport, False)
sReport &= vbCrLf
.CalculaSimbol(oConfiguracio, .Dies(i).Hores(j), .VariablesPG, .DifHoraLocalUTC, .Alçada, i - 1, moDB, iHoresRestar, sReport, False, True)
sReport &= vbCrLf
End With
我试过这样使用它,但它似乎不起作用。我猜是关于 &=
运算符的问题,或者是附加错误。
修改后的代码:
Dim sReport As new StringBuilder
Try
iHoresRestar = dtDataElaboracio.Hour
' Per cada dia, calculem els valors de les tres franges i totes les hores...
For i As Integer = 1 To oPuntInterpretar.Dies.Count
' DGARCIA: Per MeteoPlay3 no ens interessa que hi hagi diferència en l'hora local UTC,
' ja que cada previsió s'ofereix des del mateir punt evaluat.
oPuntInterpretar.DifHoraLocalUTC = 0
If (oConfiguracio.Opcions_Meteo3D.LogPGs = True) Then
' Dades del report...
sReport.Append("ID_PUNTGEOGRAFIC;" & oPuntInterpretar.IPuntGeografic & vbCrLf & _
"LATITUD_PUNTGEOGRAFIC;" & Math.Round(oPuntInterpretar.Latitud, 2) & vbCrLf & _
"LONGITUD_PUNTGEOGRAFIC;" & Math.Round(oPuntInterpretar.Longitud, 2) & vbCrLf & _
"ALCADA_PUNTGEOGRAFIC (metres);" & oPuntInterpretar.Alçada & vbCrLf & _
"TIPUS_PUNTGEOGRAFIC;" & IIf(oPuntInterpretar.Tipus = clsPuntGeografic.TipusPunt.Poblacio, "POBLACIO", "MUNTANYA") & vbCrLf & _
"DIF_HORALOCAL_UTC;" & oPuntInterpretar.DifHoraLocalUTC & vbCrLf & _
"DATA_MODEL;" & dtDataElaboracio.ToString & vbCrLf & _
"DATA_VALIDESA;" & oPuntInterpretar.Dies(i).DataValidesa.ToString & vbCrLf & vbCrLf)
End If
' a + a + de les hores calculem TMax i TMin
oPuntInterpretar.CalculaTempMin(oConfiguracio, oPuntInterpretar.Dies(i).FranjaMati, oPuntInterpretar.VariablesPG, oPuntInterpretar.DifHoraLocalUTC, oPuntInterpretar.Alçada, i - 1, iHoresRestar, sReport.ToString)
sReport.AppendLine()
oPuntInterpretar.CalculaTempMax(oConfiguracio, oPuntInterpretar.Dies(i).FranjaMati, oPuntInterpretar.VariablesPG, oPuntInterpretar.DifHoraLocalUTC, oPuntInterpretar.Alçada, i - 1, iHoresRestar, sReport.ToString)
sReport.AppendLine()
更新------------------------
这是将 sReport 传递给它的函数之一:
Public Function CalculaVentDir(ByVal config As clsConfiguracio, _
ByRef unaHora As clsHora, _
ByVal colVars As clsVariables, _
ByVal DifHoraLocalUTC As Integer, _
ByVal Alçada As Double, _
ByVal difDies As Integer, _
ByVal horaModel As Integer, _
ByRef strReport As String, _
Optional ByVal UtilitzemDifHoraLocalUTC As Boolean = True _
) As Boolean
Dim U, V As Double
Dim angle As Double
Dim horaIni, posU, posV As Integer
Dim ventDir As Double
Dim strMsg As String
Dim strTmp(7) As String
Try
If (colVars.CercaVariable(New clsVariable(33, 105, 10), posU) = True) And _
(colVars.CercaVariable(New clsVariable(34, 105, 10), posV) = True) _
Then
If (UtilitzemDifHoraLocalUTC = True) Then
horaIni = unaHora.tipusHora - DifHoraLocalUTC + difDies * 24 - horaModel
Else
horaIni = unaHora.tipusHora + difDies * 24 - horaModel
End If
strTmp(0) = "DIRECCIO DEL VENT"
strTmp(1) = "Hora local"
strTmp(2) = "Hora UTC"
strTmp(3) = "Component U"
strTmp(4) = "Component V"
strTmp(5) = "Angle exacte"
strTmp(6) = "Angle segons quadrant"
strTmp(7) = "Angle aproximat"
strMsg = "|Hora=" & unaHora.tipusHora & "|difDies=" & difDies & "|difHoraLocalUTC=" & DifHoraLocalUTC & "|horaModel=" & horaModel & "|horaIni=" & horaIni
strTmp(1) &= ";" & unaHora.tipusHora
strTmp(2) &= ";" & horaIni
If (horaIni < 0) Then
If (config.Opcions_Meteo3D.LogPGs = True) Then
RaiseEvent GetWarning("MESSAGE=No hem calculat la VentDir perquè la hora està fora dels marges" & strMsg & "|FUNCIO=CalculaVentDir_Hora", Me)
End If
unaHora.VentDir = clsConfiguracio.NO_DADA_DOUBLE
CalculaVentDir = False
End If
U = colVars.Valors(posU - 1, horaIni)
V = colVars.Valors(posV - 1, horaIni)
strTmp(3) &= ";" & U
strTmp(4) &= ";" & V
If (U = clsConfiguracio.NO_DADA_DOUBLE) Or (V = clsConfiguracio.NO_DADA_DOUBLE) Then
If (config.Opcions_Meteo3D.LogPGs = True) Then
RaiseEvent GetWarning("MESSAGE=No Assignem VentDir perquè no hem trobat dades de la hora " & horaIni & strMsg & "|FUNCIO=CalculaVentDir_Hora", Me)
End If
unaHora.VentDir = clsConfiguracio.NO_DADA_DOUBLE
CalculaVentDir = False
Else
angle = Math.Atan2(Math.Abs(V), Math.Abs(U)) * 180 / Math.PI
strTmp(5) &= ";" & angle
If (U = 0) Then
If (V >= 0) Then ventDir = 180
If (V < 0) Then ventDir = 0
Else
If (U > 0) And (V > 0) Then ventDir = 270 - angle
If (U < 0) And (V < 0) Then ventDir = 90 - angle
If (U > 0) And (V < 0) Then ventDir = 270 + angle
If (U < 0) And (V > 0) Then ventDir = 90 + angle
End If
strTmp(6) &= ";" & angle
unaHora.VentDir = ventDir
strTmp(7) &= ";" & angle
CalculaVentDir = True
End If
For j As Integer = 0 To UBound(strTmp)
strReport &= strTmp(j) & vbCrLf
Next
Else
If (config.Opcions_Meteo3D.LogPGs = True) Then
RaiseEvent GetWarning("MESSAGE=No Assignem VentDir perquè no hem trobat les variables 33 o 34|FUNCIO=CalculaVentDir_Hora", Me)
End If
CalculaVentDir = False
End If
Catch ex As Exception
If (config.Opcions_Meteo3D.LogPGs = True) Then
RaiseEvent GetError("MESSAGE=" & ex.Message & ", SOURCE=" & ex.StackTrace & ", FUNCIO=CalculaVentDir_Hora", Me)
End If
unaHora.VentDir = clsConfiguracio.NO_DADA_DOUBLE
CalculaVentDir = False
End Try
End Function
问题出在这些行中:
oPuntInterpretar.CalculaTempMin(oConfiguracio, oPuntInterpretar.Dies(i).FranjaMati, oPuntInterpretar.VariablesPG, oPuntInterpretar.DifHoraLocalUTC, oPuntInterpretar.Alçada, i - 1, iHoresRestar, sReport.ToString)
oPuntInterpretar.CalculaTempMax(oConfiguracio, oPuntInterpretar.Dies(i).FranjaMati, oPuntInterpretar.VariablesPG, oPuntInterpretar.DifHoraLocalUTC, oPuntInterpretar.Alçada, i - 1, iHoresRestar, sReport.ToString)
... 类似的,你将 sReport.ToString()
传递给一个方法。
您需要将这些功能更改为甚至不要求报告。如果他们需要向报告中添加一些内容,请让他们 return 一个字符串,然后在调用方法时附加结果:
sReport.Appdend(oPuntInterpretar.CalculaTempMin(oConfiguracio, oPuntInterpretar.Dies(i).FranjaMati, oPuntInterpretar.VariablesPG, oPuntInterpretar.DifHoraLocalUTC, oPuntInterpretar.Alçada, i - 1, iHoresRestar))
您也可以更改方法以改为接受 StringBuilder 对象,但这种方式是更好的编码风格。该方法并不真正需要知道报告字符串中已有的内容,函数的参数超过 3 或 4 个都是代码异味。
如果此报告很大,即使让 StringBuilder 正常工作也可能无法解决您的 OutOfMemory 问题。如果问题仍然存在,下一个要查看的地方是使用 Stream
,例如 FileStream
、ASP.Net ReponseStream
,或 Console.Out
流,所有这些都可以通过 TextWriter
.
最后,我们可以改进初始的 StringBuilder 翻译:
Dim reportBase As String = "ID_PUNTGEOGRAFIC;{0}{1}LATITUD_PUNTGEOGRAFIC;{2}{1}LONGITUD_PUNTGEOGRAFIC;{3}{1}ALCADA_PUNTGEOGRAFIC (metres);{4}{1}TIPUS_PUNTGEOGRAFIC;{5}{1}DIF_HORALOCAL_UTC;{6}{1}DATA_MODEL;{7}{1}DATA_VALIDESA;{8}{1}{1}"
Dim Report As New StringBuilder()
iHoresRestar = dtDataElaboracio.Hour
For Each die In oPuntInterpretar.Dies
If oConfiguracio.Opcions_Meteo3D.LogPGs Then
Report.AppendFormat(reportBase, _
oPuntInterpretar.IPuntGeografic,vbCrLf, _
Math.Round(oPuntInterpretar.Latitud, 2), _
Math.Round(oPuntInterpretar.Longitud, 2), _
oPuntInterpretar.Alçada, _
IIf(oPuntInterpretar.Tipus = clsPuntGeografic.TipusPunt.Poblacio, "POBLACIO", "MUNTANYA"), _
oPuntInterpretar.DifHoraLocalUTC, _
dtDataElaboracio.ToString, _
die.DataValidesa.ToString )
End If
'...
如果您能够更新到 Visual Studio 的更新版本(现在在大多数情况下 免费 ),我们可以进一步改进代码。我现在将特别添加 Microsoft 建议 反对 在变量名称上使用匈牙利符号疣。现在只是 report
,而不是 sReport
,puntInterpretar
而不是 oPuntInterpretar
。