2009-08-14 14 views
8

Utilizzo il controllo WebBrowser in .Net per eseguire conversioni di marketing di affiliazione di terze parti.Caching e controllo WebBrowser in .Net

Ho una tabella di attesa in un database con tutti gli script/immagini da eseguire. Eseguo il ciclo di tutte queste operazioni in un'app WinForms con il controllo WebBrowser. Dopo aver eseguito uno script/immagine I Dispose il controllo WebBrowser, impostarlo su null e lo rinnova con una nuova istanza di controllo WebBrowser.

Considerate questo URL: visualizza http://renderserver/RenderScript.aspx?id=1

RenderScript.aspx di un'immagine con un URL di esempio: http://3rdparty/img.ashx?id=9343

Io uso Fiddler per vedere tutte le richieste e le risposte, e quando lo stesso URL viene eseguito due volte, usa un qualche tipo di cache. Quella cache esiste sotto il controllo WebBrowser stesso.

Questa cache indica che img.ashx non viene chiamato.

Ho provato a utilizzare Internet Explorer per richiedere l'URL: http://renderserver/RenderScript.aspx?id=1 e premere F5. Quindi è richiesto perfettamente.

Ma se faccio clic sulla barra degli indirizzi e preme Invio per navigare di nuovo verso lo stesso URL - non è richiesto. Quando utilizzo Firefox, richiederò la pagina e l'immagine in qualsiasi momento, indipendentemente dall'utilizzo di F5 o dalla barra degli indirizzi.

Ho trovato alcune chiamate API Win32 (http://support.microsoft.com/kb/326201) che è stato in grado di cancellare la cache. Ha funzionato sulla mia macchina locale. Quindi l'app è stata distribuita su un server con Windows Server 2003 Standard x64 (il mio computer è Vista x86).

E ora l'API chiama per cancellare la cache non funziona.

Qualche idea sul perché le chiamate API non funzionano su Windows Server, ma funziona su Vista? Entrambe le macchine con IE8.

risposta

4

ho avuto la s problema (abbastanza) un po 'di tempo fa. Microsoft ha una pagina che è stato molto utile con questo:

http://support.microsoft.com/default.aspx?scid=http://support.microsoft.com:80/support/kb/articles/q326/2/01.asp&NoWebContent=1

ho creato una classe fuori dal campione di Microsoft, ma ho anche dovuto aggiungere un paio se le dichiarazioni di smettere di elaborazione quando non ci sono più elementi; è passato un po 'di tempo, ma sono abbastanza sicuro che genererebbe un errore (vedi ERROR_NO_MORE_ITEMS nel codice qui sotto).

Spero sia utile!

using System; 
using System.Runtime.InteropServices; 

// copied from: http://support.microsoft.com/default.aspx?scid=http://support.microsoft.com:80/support/kb/articles/q326/2/01.asp&NoWebContent=1 

namespace PowerCode 
{ 
    public class IECache 
    { 
     // For PInvoke: Contains information about an entry in the Internet cache 
     [StructLayout(LayoutKind.Explicit, Size = 80)] 
     public struct INTERNET_CACHE_ENTRY_INFOA 
     { 
      [FieldOffset(0)] 
      public uint dwStructSize; 
      [FieldOffset(4)] 
      public IntPtr lpszSourceUrlName; 
      [FieldOffset(8)] 
      public IntPtr lpszLocalFileName; 
      [FieldOffset(12)] 
      public uint CacheEntryType; 
      [FieldOffset(16)] 
      public uint dwUseCount; 
      [FieldOffset(20)] 
      public uint dwHitRate; 
      [FieldOffset(24)] 
      public uint dwSizeLow; 
      [FieldOffset(28)] 
      public uint dwSizeHigh; 
      [FieldOffset(32)] 
      public FILETIME LastModifiedTime; 
      [FieldOffset(40)] 
      public FILETIME ExpireTime; 
      [FieldOffset(48)] 
      public FILETIME LastAccessTime; 
      [FieldOffset(56)] 
      public FILETIME LastSyncTime; 
      [FieldOffset(64)] 
      public IntPtr lpHeaderInfo; 
      [FieldOffset(68)] 
      public uint dwHeaderInfoSize; 
      [FieldOffset(72)] 
      public IntPtr lpszFileExtension; 
      [FieldOffset(76)] 
      public uint dwReserved; 
      [FieldOffset(76)] 
      public uint dwExemptDelta; 
     } 

     // For PInvoke: Initiates the enumeration of the cache groups in the Internet cache 
     [DllImport(@"wininet", SetLastError = true, CharSet = CharSet.Auto, EntryPoint = "FindFirstUrlCacheGroup", CallingConvention = CallingConvention.StdCall)] 
     public static extern IntPtr FindFirstUrlCacheGroup(int dwFlags, int dwFilter, IntPtr lpSearchCondition, int dwSearchCondition, ref long lpGroupId, IntPtr lpReserved); 

     // For PInvoke: Retrieves the next cache group in a cache group enumeration 
     [DllImport(@"wininet", SetLastError = true, CharSet = CharSet.Auto, EntryPoint = "FindNextUrlCacheGroup", CallingConvention = CallingConvention.StdCall)] 
     public static extern bool FindNextUrlCacheGroup(IntPtr hFind, ref long lpGroupId, IntPtr lpReserved); 

     // For PInvoke: Releases the specified GROUPID and any associated state in the cache index file 
     [DllImport(@"wininet", SetLastError = true, CharSet = CharSet.Auto, EntryPoint = "DeleteUrlCacheGroup", CallingConvention = CallingConvention.StdCall)] 
     public static extern bool DeleteUrlCacheGroup(long GroupId, int dwFlags, IntPtr lpReserved); 

     // For PInvoke: Begins the enumeration of the Internet cache 
     [DllImport(@"wininet", SetLastError = true, CharSet = CharSet.Auto, EntryPoint = "FindFirstUrlCacheEntryA", CallingConvention = CallingConvention.StdCall)] 
     public static extern IntPtr FindFirstUrlCacheEntry([MarshalAs(UnmanagedType.LPTStr)] string lpszUrlSearchPattern, IntPtr lpFirstCacheEntryInfo, ref int lpdwFirstCacheEntryInfoBufferSize); 

     // For PInvoke: Retrieves the next entry in the Internet cache 
     [DllImport(@"wininet", SetLastError = true, CharSet = CharSet.Auto, EntryPoint = "FindNextUrlCacheEntryA", CallingConvention = CallingConvention.StdCall)] 
     public static extern bool FindNextUrlCacheEntry(IntPtr hFind, IntPtr lpNextCacheEntryInfo, ref int lpdwNextCacheEntryInfoBufferSize); 

     // For PInvoke: Removes the file that is associated with the source name from the cache, if the file exists 
     [DllImport(@"wininet", SetLastError = true, CharSet = CharSet.Auto, EntryPoint = "DeleteUrlCacheEntryA", CallingConvention = CallingConvention.StdCall)] 
     public static extern bool DeleteUrlCacheEntry(IntPtr lpszUrlName); 

     public static void ClearCache() 
     { 
      // Indicates that all of the cache groups in the user's system should be enumerated 
      const int CACHEGROUP_SEARCH_ALL = 0x0; 
      // Indicates that all the cache entries that are associated with the cache group 
      // should be deleted, unless the entry belongs to another cache group. 
      const int CACHEGROUP_FLAG_FLUSHURL_ONDELETE = 0x2; 
      // File not found. 
      const int ERROR_FILE_NOT_FOUND = 0x2; 
      // No more items have been found. 
      const int ERROR_NO_MORE_ITEMS = 259; 
      // Pointer to a GROUPID variable 
      long groupId = 0; 

      // Local variables 
      int cacheEntryInfoBufferSizeInitial = 0; 
      int cacheEntryInfoBufferSize = 0; 
      IntPtr cacheEntryInfoBuffer = IntPtr.Zero; 
      INTERNET_CACHE_ENTRY_INFOA internetCacheEntry; 
      IntPtr enumHandle = IntPtr.Zero; 
      bool returnValue = false; 

      // Delete the groups first. 
      // Groups may not always exist on the system. 
      // For more information, visit the following Microsoft Web site: 
      // http://msdn.microsoft.com/library/?url=/workshop/networking/wininet/overview/cache.asp    
      // By default, a URL does not belong to any group. Therefore, that cache may become 
      // empty even when the CacheGroup APIs are not used because the existing URL does not belong to any group.    
      enumHandle = FindFirstUrlCacheGroup(0, CACHEGROUP_SEARCH_ALL, IntPtr.Zero, 0, ref groupId, IntPtr.Zero); 

      // If there are no items in the Cache, you are finished. 
      if (enumHandle != IntPtr.Zero && ERROR_NO_MORE_ITEMS == Marshal.GetLastWin32Error()) { 
       return; 
      } 

      // Loop through Cache Group, and then delete entries. 
      while (true) { 
       if (ERROR_NO_MORE_ITEMS == Marshal.GetLastWin32Error() || ERROR_FILE_NOT_FOUND == Marshal.GetLastWin32Error()) { 
        break; 
       } 

       // Delete a particular Cache Group. 
       returnValue = DeleteUrlCacheGroup(groupId, CACHEGROUP_FLAG_FLUSHURL_ONDELETE, IntPtr.Zero); 
       if (!returnValue && ERROR_FILE_NOT_FOUND == Marshal.GetLastWin32Error()) { 
        returnValue = FindNextUrlCacheGroup(enumHandle, ref groupId, IntPtr.Zero); 
       } 

       if (!returnValue && (ERROR_NO_MORE_ITEMS == Marshal.GetLastWin32Error() || ERROR_FILE_NOT_FOUND == Marshal.GetLastWin32Error())) { 
        break; 
       } 
      } 

      // Start to delete URLs that do not belong to any group. 
      enumHandle = FindFirstUrlCacheEntry(null, IntPtr.Zero, ref cacheEntryInfoBufferSizeInitial); 
      if (enumHandle != IntPtr.Zero && ERROR_NO_MORE_ITEMS == Marshal.GetLastWin32Error()) { 
       return; 
      } 

      cacheEntryInfoBufferSize = cacheEntryInfoBufferSizeInitial; 
      cacheEntryInfoBuffer = Marshal.AllocHGlobal(cacheEntryInfoBufferSize); 
      enumHandle = FindFirstUrlCacheEntry(null, cacheEntryInfoBuffer, ref cacheEntryInfoBufferSizeInitial); 

      while (true) { 
       internetCacheEntry = (INTERNET_CACHE_ENTRY_INFOA)Marshal.PtrToStructure(cacheEntryInfoBuffer, typeof(INTERNET_CACHE_ENTRY_INFOA)); 

       if (ERROR_NO_MORE_ITEMS == Marshal.GetLastWin32Error()) { 
        break; 
       } 

       cacheEntryInfoBufferSizeInitial = cacheEntryInfoBufferSize; 
       returnValue = DeleteUrlCacheEntry(internetCacheEntry.lpszSourceUrlName); 
       if (!returnValue) { 
        returnValue = FindNextUrlCacheEntry(enumHandle, cacheEntryInfoBuffer, ref cacheEntryInfoBufferSizeInitial); 
       } 
       if (!returnValue && ERROR_NO_MORE_ITEMS == Marshal.GetLastWin32Error()) { 
        break; 
       } 
       if (!returnValue && cacheEntryInfoBufferSizeInitial > cacheEntryInfoBufferSize) { 
        cacheEntryInfoBufferSize = cacheEntryInfoBufferSizeInitial; 
        cacheEntryInfoBuffer = Marshal.ReAllocHGlobal(cacheEntryInfoBuffer, (IntPtr)cacheEntryInfoBufferSize); 
        returnValue = FindNextUrlCacheEntry(enumHandle, cacheEntryInfoBuffer, ref cacheEntryInfoBufferSizeInitial); 
       } 
      } 
      Marshal.FreeHGlobal(cacheEntryInfoBuffer); 
     } 
    } 
} 

di usarlo nel codice, basta chiamare:

IECache.ClearCache() 

prima di chiamare i metodi navigare.

0

Fiddler utilizza fondamentalmente lo stesso codice dell'articolo KB per svuotare la cache di WinINET e lo uso su Win2k3 ogni giorno.

Invece di cancellare l'intera cache dell'utente, la correzione corretta consiste nell'impostare l'intestazione di risposta HTTP appropriata per vietare la memorizzazione nella cache. È possibile ottenere ulteriori informazioni sulla memorizzazione nella cache di WinINET qui: http://www.enhanceie.com/redir/?id=httpperf

(In alternativa, è possibile semplicemente aggiungere un parametro di stringa di query randomizzato, in questo modo, ogni volta che il controllo rileva una richiesta per la risorsa, l'URL è diverso e la cache è quindi automaticamente esclusa)

+0

Negativo. Non ha nulla a che fare con le intestazioni della cache. Sono già impostati correttamente per evitare il caching. E altri browser non fanno il caching, come ho sottolineato. – MartinHN

+0

E il parametro randomizzato non funziona neanche. Perché l'HTML di terze parti che viene caricato dalla pagina ScriptRender è completamente fuori dal mio controllo. Quindi posso solo aggiungere il parametro alla pagina RenderScript. Il tag immagine (a volte IFrame, a volte javascript), viene caricato con l'URL statico. – MartinHN

+0

Mi spiace, ma ti sbagli. WinINET/IE/WebOCs non riutilizzeranno le risposte memorizzate nella cache se sono impostate le intestazioni di risposta appropriate. Quali sono le intestazioni esatte inviate da ASHX? Potete inviarmi un Network Capture (www.fiddlercap.com)? – EricLaw

-1

questo dovrebbe fare il trucco:.

Response.Cache.SetCacheability(HttpCacheability.NoCache); 
+1

-1: non sta parlando di ASP.NET. Sta parlando di IE. –

0

Prova questa ...

[DllImport("wininet.dll", SetLastError = true)] 
     private static extern bool InternetSetOption(IntPtr hInternet, int dwOption, IntPtr lpBuffer, int lpdwBufferLength); 
private const int INTERNET_OPTION_END_BROWSER_SESSION = 42; 
private void clearCache() 
{ 
    try 
    { 
     Utilities.Web.WebBrowserHelper.WebBrowserHelper.ClearCache(); 
     InternetSetOption(IntPtr.Zero, INTERNET_OPTION_END_BROWSER_SESSION, IntPtr.Zero, 0); 
    } 
    catch (Exception exception) 
    { 
     //throw; 
    } 

} 
+1

Dov'è 'Utilities.WebBrowserHelper.WebBrowserHelper.ClearCache();' definito? – Nate

0

Questo articolo kb a cui tutti ci colleghiamo ha numerosi errori (da cui proviene il codice sorgente della risposta selezionata) e ho perso ~ 2 giorni cercando di farlo funzionare in tutte le impostazioni necessarie. È copiata incollata su Internet, e ci sono numerosi bug segnalati in base alla versione di IE e di OS.

Fiddler è stato originariamente scritto da un dipendente Microsoft ed è alimentato da FiddlerCore.dll. Telerik (gli attuali proprietari/manutentori/venditori) di Fiddler aggiornano, mantengono e regalano FiddlerCore gratuitamente. Se non vuoi aggiungere un riferimento a FiddlerCore, puoi disassemblare la dll e mostra il modo CORRETTO per chiamare tutte queste funzioni WinINet orribilmente documentate, ma penso che postarla qui sarebbe un disservizio per Telerik/plagio.

Attualmente, Fiddlercore è ospitato qui: http://www.telerik.com/fiddler/fiddlercore

0

Il codice originale da https://support.microsoft.com/en-us/kb/326201 sembra buggy

controllando la documentazione MSDN e anche la versione VB qui: https://support.microsoft.com/en-us/kb/262110

ho modificato il codice come questo e ora perché funziona per me (il problema era nell'esecuzione di FindNextUrlCacheGroup e FindNextUrlCacheEntry):

using System; 
using System.Runtime.InteropServices; 

namespace Q326201CS 
{ 
    // Class for deleting the cache. 
    public class DeleteIECache 
    { 
     // For PInvoke: Contains information about an entry in the Internet cache 
     [StructLayout(LayoutKind.Explicit, Size=80)] 
     public struct INTERNET_CACHE_ENTRY_INFOA 
     { 
      [FieldOffset(0)] public uint dwStructSize; 
      [FieldOffset(4)] public IntPtr lpszSourceUrlName; 
      [FieldOffset(8)] public IntPtr lpszLocalFileName; 
      [FieldOffset(12)] public uint CacheEntryType; 
      [FieldOffset(16)] public uint dwUseCount; 
      [FieldOffset(20)] public uint dwHitRate; 
      [FieldOffset(24)] public uint dwSizeLow; 
      [FieldOffset(28)] public uint dwSizeHigh; 
      [FieldOffset(32)] public FILETIME LastModifiedTime; 
      [FieldOffset(40)] public FILETIME ExpireTime; 
      [FieldOffset(48)] public FILETIME LastAccessTime; 
      [FieldOffset(56)] public FILETIME LastSyncTime; 
      [FieldOffset(64)] public IntPtr lpHeaderInfo; 
      [FieldOffset(68)] public uint dwHeaderInfoSize; 
      [FieldOffset(72)] public IntPtr lpszFileExtension; 
      [FieldOffset(76)] public uint dwReserved; 
      [FieldOffset(76)] public uint dwExemptDelta; 
     } 

     // For PInvoke: Initiates the enumeration of the cache groups in the Internet cache 
     [DllImport(@"wininet", 
      SetLastError=true, 
      CharSet=CharSet.Auto, 
      EntryPoint="FindFirstUrlCacheGroup", 
      CallingConvention=CallingConvention.StdCall)] 
     public static extern IntPtr FindFirstUrlCacheGroup(
      int dwFlags, 
      int dwFilter, 
      IntPtr lpSearchCondition, 
      int dwSearchCondition, 
      ref long lpGroupId, 
      IntPtr lpReserved); 

     // For PInvoke: Retrieves the next cache group in a cache group enumeration 
     [DllImport(@"wininet", 
      SetLastError=true, 
      CharSet=CharSet.Auto, 
      EntryPoint="FindNextUrlCacheGroup", 
      CallingConvention=CallingConvention.StdCall)] 
     public static extern bool FindNextUrlCacheGroup(
      IntPtr hFind, 
      ref long lpGroupId, 
      IntPtr lpReserved); 

     // For PInvoke: Releases the specified GROUPID and any associated state in the cache index file 
     [DllImport(@"wininet", 
      SetLastError=true, 
      CharSet=CharSet.Auto, 
      EntryPoint="DeleteUrlCacheGroup", 
      CallingConvention=CallingConvention.StdCall)] 
     public static extern bool DeleteUrlCacheGroup(
      long GroupId, 
      int dwFlags, 
      IntPtr lpReserved); 

     // For PInvoke: Begins the enumeration of the Internet cache 
     [DllImport(@"wininet", 
      SetLastError=true, 
      CharSet=CharSet.Auto, 
      EntryPoint="FindFirstUrlCacheEntryA", 
      CallingConvention=CallingConvention.StdCall)] 
     public static extern IntPtr FindFirstUrlCacheEntry(
      [MarshalAs(UnmanagedType.LPTStr)] string lpszUrlSearchPattern, 
      IntPtr lpFirstCacheEntryInfo, 
      ref int lpdwFirstCacheEntryInfoBufferSize); 

     // For PInvoke: Retrieves the next entry in the Internet cache 
     [DllImport(@"wininet", 
      SetLastError=true, 
      CharSet=CharSet.Auto, 
      EntryPoint="FindNextUrlCacheEntryA", 
      CallingConvention=CallingConvention.StdCall)] 
     public static extern bool FindNextUrlCacheEntry(
      IntPtr hFind, 
      IntPtr lpNextCacheEntryInfo, 
      ref int lpdwNextCacheEntryInfoBufferSize); 

     // For PInvoke: Removes the file that is associated with the source name from the cache, if the file exists 
     [DllImport(@"wininet", 
      SetLastError=true, 
      CharSet=CharSet.Auto, 
      EntryPoint="DeleteUrlCacheEntryA", 
      CallingConvention=CallingConvention.StdCall)] 
     public static extern bool DeleteUrlCacheEntry(
      IntPtr lpszUrlName); 


     public static void doDelete() 
     { 
      // Indicates that all of the cache groups in the user's system should be enumerated 
      const int CACHEGROUP_SEARCH_ALL = 0x0; 
      // Indicates that all the cache entries that are associated with the cache group 
      // should be deleted, unless the entry belongs to another cache group. 
      const int CACHEGROUP_FLAG_FLUSHURL_ONDELETE = 0x2; 
      // File not found. 
      const int ERROR_FILE_NOT_FOUND = 0x2; 
      // No more items have been found. 
      const int ERROR_NO_MORE_ITEMS = 259; 
      // Pointer to a GROUPID variable 
      long groupId = 0; 

      // Local variables 
      int cacheEntryInfoBufferSizeInitial = 0; 
      int cacheEntryInfoBufferSize = 0; 
      IntPtr cacheEntryInfoBuffer = IntPtr.Zero; 
      INTERNET_CACHE_ENTRY_INFOA internetCacheEntry; 
      IntPtr enumHandle = IntPtr.Zero; 
      bool returnValue = false; 

      // Delete the groups first. 
      // Groups may not always exist on the system. 
      // For more information, visit the following Microsoft Web site: 
      // http://msdn.microsoft.com/library/?url=/workshop/networking/wininet/overview/cache.asp   
      // By default, a URL does not belong to any group. Therefore, that cache may become 
      // empty even when the CacheGroup APIs are not used because the existing URL does not belong to any group.   
      enumHandle = FindFirstUrlCacheGroup(0, CACHEGROUP_SEARCH_ALL, IntPtr.Zero, 0, ref groupId, IntPtr.Zero); 
      // If there are no items in the Cache, you are finished. 
      if (enumHandle != IntPtr.Zero && ERROR_NO_MORE_ITEMS == Marshal.GetLastWin32Error()) 
       return; 

      // Loop through Cache Group, and then delete entries. 
      while(true) 
      { 
       if (ERROR_NO_MORE_ITEMS == Marshal.GetLastWin32Error() || ERROR_FILE_NOT_FOUND == Marshal.GetLastWin32Error()) 
       { 
        break; 
       } 

       // Delete a particular Cache Group. 
       returnValue = DeleteUrlCacheGroup(groupId, CACHEGROUP_FLAG_FLUSHURL_ONDELETE, IntPtr.Zero); 
       //if (returnValue || (!returnValue && ERROR_FILE_NOT_FOUND == Marshal.GetLastWin32Error())) 
       //{ 
        returnValue = FindNextUrlCacheGroup(enumHandle, ref groupId, IntPtr.Zero); 
       //} 

       if (!returnValue && (ERROR_NO_MORE_ITEMS == Marshal.GetLastWin32Error() || ERROR_FILE_NOT_FOUND == Marshal.GetLastWin32Error())) 
        break; 
      } 

      // Start to delete URLs that do not belong to any group. 
      enumHandle = FindFirstUrlCacheEntry(null, IntPtr.Zero, ref cacheEntryInfoBufferSizeInitial); 
      if (enumHandle == IntPtr.Zero && ERROR_NO_MORE_ITEMS == Marshal.GetLastWin32Error()) 
       return; 

      cacheEntryInfoBufferSize = cacheEntryInfoBufferSizeInitial; 
      cacheEntryInfoBuffer = Marshal.AllocHGlobal(cacheEntryInfoBufferSize); 
      enumHandle = FindFirstUrlCacheEntry(null, cacheEntryInfoBuffer, ref cacheEntryInfoBufferSizeInitial); 

      while(true) 
      { 
       internetCacheEntry = (INTERNET_CACHE_ENTRY_INFOA)Marshal.PtrToStructure(cacheEntryInfoBuffer, typeof(INTERNET_CACHE_ENTRY_INFOA)); 

       if (ERROR_NO_MORE_ITEMS == Marshal.GetLastWin32Error()) 
       { 
        break; 
       } 

       cacheEntryInfoBufferSizeInitial = cacheEntryInfoBufferSize; 
       returnValue = DeleteUrlCacheEntry(internetCacheEntry.lpszSourceUrlName);     
       //if (!returnValue) 
       //{ 
        returnValue = FindNextUrlCacheEntry(enumHandle, cacheEntryInfoBuffer, ref cacheEntryInfoBufferSizeInitial); 
       //} 
       if (!returnValue && ERROR_NO_MORE_ITEMS == Marshal.GetLastWin32Error()) 
       { 
        break; 
       }   
       if (!returnValue && cacheEntryInfoBufferSizeInitial > cacheEntryInfoBufferSize) 
       { 
        cacheEntryInfoBufferSize = cacheEntryInfoBufferSizeInitial; 
        cacheEntryInfoBuffer = Marshal.ReAllocHGlobal(cacheEntryInfoBuffer, (IntPtr) cacheEntryInfoBufferSize); 
        returnValue = FindNextUrlCacheEntry(enumHandle, cacheEntryInfoBuffer, ref cacheEntryInfoBufferSizeInitial);     
       } 
      } 
      Marshal.FreeHGlobal(cacheEntryInfoBuffer);  
     } 
    } 
} 
Problemi correlati