我正在使用PrivateFontCollection在我的网络服务器上安装上传的字体。下面的代码有效,但在第二次上传字体时,PrivateFontCollection引用上传的第一个字体。所以有些东西没有被正确处理掉。有人发现我做错了吗?
var fontName = string.Empty;
using (var ms = new MemoryStream(fontBytes))
{
// used to store our font and make it available in our app
using (var pfc = new PrivateFontCollection())
{
//create memory pointer
IntPtr data = Marshal.AllocCoTaskMem((int)ms.Length);
try
{
//copy the bytes to the unsafe memory block
Marshal.Copy(fontBytes, 0, data, (int)ms.Length);
// We HAVE to do this to register the font to the system (Weird .NET bug !)
uint cFonts = 0;
AddFontMemResourceEx(data, (uint)fontBytes.Length, IntPtr.Zero, ref cFonts);
//pass the font to the font collection
pfc.AddMemoryFont(data, (int)ms.Length);
var fontWithMime = "data:application/x-font-truetype;charset=utf-8;base64," + cleanFontData;
fontName = pfc.Families[0].Name;
//db work here
}
finally
{
ms.Close();
Marshal.FreeCoTaskMem(data);
}
}
}
答案 0 :(得分:2)
PrivateFontCollection是一个非常有缺陷的类,你必须非常小心地使用它。您现有代码中的一个非常严重的错误是Marshal.FreeCoTaskMem()调用。您需要确保不调用此函数,直到 代码停止使用您从该系列创建的任何字体对象为止。如果不这样做会导致随机字形损坏,如果运气好,您只会获得AccessViolationException。基本问题是字体将继续使用您使用AllocCoTaskMem()分配的内存,它完全不知道内存不再有效。重新使用内存时会发生损坏。
此外,虽然该类具有AddMemoryFont()方法,但它没有相应的RemoveMemoryFont()方法。清理的唯一方法是调用PrivateFontCollection.Dispose()。这将删除集合中所有字体。使用与上一段相同的规定,只有在确定不再使用任何Font对象时,才能调用Dispose()。过早调用不会导致异常。
非常尴尬的行为,只有真正安全的使用PFC的方法是在应用程序的生命周期中保持它。当然,在网络应用程序中非常痛苦。
您可以假设添加的字体是FontFamily []数组中的 last 。不是第一个,因为你已经实现了它。
答案 1 :(得分:1)
根据AddFontMemResourceEx function:
要删除已安装的字体,请调用RemoveFontMemResourceEx。 但是,当进程消失时,系统将卸载字体 即使进程没有调用RemoveFontMemResource。
但是,我不认为你这样做。这可能是原因。