.NET PrivateFontCollection - как снять блокировку файла, когда закончите
У меня есть функция, которая возвращает PrivateFontCollection:
Public Shared Function GetCustomFonts() As PrivateFontCollection
Dim result = New PrivateFontCollection
Dim customFontFiles = {"Garamond.TTF", "Garamond-Bold.TTF", "Garamond-Italic.TTF", "EurostileExtended-Roman-DTC.TTF"}
For Each fontFile In customFontFiles
result.AddFontFile(Hosting.HostingEnvironment.MapPath("/Includes/" & fontFile))
Next
Return result
End Function
Затем я использую функцию следующим образом:
Using customFonts = Common.GetCustomFonts()
' Do some stuff here
End Using
Я ожидаю, что файлы будут выпущены, но они все еще заблокированы: я получаю следующую ошибку: "Действие не может быть завершено, потому что файл открыт в системе. Закройте файл и попробуйте снова.
Закрытие сайта в IIS не помогает; мы должны переработать пул приложений для его выпуска.
Кто-нибудь может посоветовать, как использовать PrivateFontCollection таким образом, чтобы файлы были выпущены между использованиями?
1 ответ
В качестве обходного пути я загрузил шрифты в память и использовал вместо него "AddMemoryFont". Смотрите код ниже. Имейте в виду, это первый раз, когда я коснулся неуправляемых ресурсов в.NET, поэтому я не могу гарантировать, что с управлением памятью ниже все в порядке.
Imports System.Drawing.Text
Imports System.Runtime.InteropServices
Public Class CustomFontService
Implements IDisposable
Dim _fontBuffers As List(Of IntPtr)
Dim _collection As PrivateFontCollection
Public Sub New()
_collection = New PrivateFontCollection
_fontBuffers = New List(Of IntPtr)
Dim customFontFiles = {"Garamond.TTF", "Garamond-Bold.TTF", "Garamond-Italic.TTF", "EurostileExtended-Roman-DTC.TTF"}
For Each fontFile In customFontFiles
Dim fontBytes = System.IO.File.ReadAllBytes(Hosting.HostingEnvironment.MapPath("/Includes/" & fontFile))
Dim fontBuffer As IntPtr = Marshal.AllocHGlobal(fontBytes.Length)
Marshal.Copy(fontBytes, 0, fontBuffer, fontBytes.Length)
_fontBuffers.Add(fontBuffer)
_collection.AddMemoryFont(fontBuffer, fontBytes.Length)
Next
End Sub
Public Function GetCustomFonts() As PrivateFontCollection
Return _collection
End Function
#Region "IDisposable Support"
Private disposedValue As Boolean ' To detect redundant calls
' IDisposable
Protected Overridable Sub Dispose(disposing As Boolean)
If Not Me.disposedValue Then
If disposing Then
' TODO: dispose managed state (managed objects).
End If
For Each buf In _fontBuffers
If (buf <> IntPtr.Zero) Then
Marshal.FreeHGlobal(buf)
End If
Next
' TODO: free unmanaged resources (unmanaged objects) and override Finalize() below.
' TODO: set large fields to null.
End If
Me.disposedValue = True
End Sub
' TODO: override Finalize() only if Dispose(ByVal disposing As Boolean) above has code to free unmanaged resources.
Protected Overrides Sub Finalize()
' Do not change this code. Put cleanup code in Dispose(ByVal disposing As Boolean) above.
Dispose(False)
MyBase.Finalize()
End Sub
' This code added by Visual Basic to correctly implement the disposable pattern.
Public Sub Dispose() Implements IDisposable.Dispose
' Do not change this code. Put cleanup code in Dispose(disposing As Boolean) above.
Dispose(True)
GC.SuppressFinalize(Me)
End Sub
#End Region
End Class