J'utilise le contrôle WebBrowser dans .NET pour exécuter quelques conversions de marketing d'affiliation tierce. p>
J'ai une table de file d'attente dans une base de données avec tous les scripts / images à exécuter. Je boucle à travers tous ceux-ci dans une application WinForms avec le contrôle WebBrowser. Après avoir exécuté un script / image, je dispose du contrôle WebBrowser, définissez-le sur NULL et la renouvelez avec une nouvelle instance de contrôle WebBrowser. P>
Considérez cette URL: http: //renderserver/renderscript.aspx? id = 1 < / p>
Renderscript.aspx Affiche une image avec une URL d'EG: http: //3rdparty/img.ashx? id = 9343 p>
J'utilise Fiddler pour voir toutes les demandes et réponses, et lorsque la même URL est exécutée deux fois, elle utilise une sorte de cache. Ce cache existe sous le contrôle WebBrowser lui-même. P>
Ce cache signifie que le img.achx n'est pas appelé. p>
J'ai essayé d'utiliser Internet Explorer pour demander l'URL: http: //renderserver/renderscript.aspx? ID = 1 et frappez F5. Alors il est demandé parfaitement. P>
Mais si je clique sur la barre d'adresse et heurtez Entrée pour naviguer à nouveau à la même URL - il n'est pas demandé. Lorsque j'utilise Firefox est demander la page et l'image à chaque fois que j'utilise F5 ou que je navigue depuis la barre d'adresse. P>
6 Réponses :
Fiddler utilise fondamentalement le même code que dans l'article de KB pour effacer le cache Wininet, et je l'utilise sur Win2k3 tous les jours. p>
Plutôt que d'essuyer le cache entier de l'utilisateur, le correctif approprié consiste à définir l'en-tête de réponse HTTP approprié pour interdire la mise en cache. Vous pouvez en apprendre davantage sur Wininet Caching ici: http://www.enhanceie.com/redir/ ? id = httpperf p>
(Alternativement, vous pouvez simplement ajouter un paramètre de chaîne de requête randomisé; de cette façon, chaque fois que la commande rencontre une demande de ressource, l'URL est différente et que le cache est donc automatiquement contourné.) P>
Négatif. Cela n'a rien à voir avec les en-têtes de cache. Ils sont déjà installés correctement pour éviter la mise en cache. Et d'autres navigateurs ne mettent pas la mise en cache, comme je l'ai signalé.
Et le paramètre randomisé ne fonctionne pas non plus. Étant donné que la 3ème partie HTML chargée de la page Scriptrender est complètement hors de mon contrôle. Je ne peux donc ajouter que le paramètre à la page Renderscript. La balise d'image (parfois iframe, parfois javascript) est chargée avec l'URL statique.
Désolé, mais vous vous trompez. Wininet / IE / Webocs ne réutilisera pas les réponses en cache si les en-têtes de réponse appropriés sont définis. Quels sont les en-têtes exacts que l'ashx envoie? Pouvez-vous m'envoyer une capture de réseau (www.fiddlerercap.com)?
Utilisation de IIS 7.5 Livrer des fichiers PDF, sans contrôle de cache, Wininet défini pour ne pas redemiler définitivement le fichier PDF lorsqu'il est plus récent et a été brisé depuis au moins 8 au moins.
@ Netmage-- vous êtes confus. Si NO i> Les en-têtes de contrôle du cache sont présents pour l'interdire, la norme spécifie que la réponse peut être mise en cache et réutilisée.
Cela devrait faire l'astuce:
Essayez ceci ...
Où est utilitaires.web.webbrowseerhelper.webbrowserhelper.tlearcache (); code> défini?
J'ai eu le même problème (tout à fait) pendant le retour. Microsoft a une page très utile avec ceci:
J'ai créé une classe de l'échantillon de Microsoft, mais j'ai également dû ajouter un couple si des déclarations pour arrêter le traitement lorsqu'il n'y a plus de éléments; Cela fait un moment maintenant, mais je suis sûr que cela lancerait une erreur (voir J'espère que c'est utile! P> < Pré> xxx pré> Pour l'utiliser dans votre code, appelez simplement: p> Avant d'appeler les méthodes de navigation. p> p> error_no_more_items code> dans le code ci-dessous). P>
Il a fait une impasse et je pense devoir changer! Retourvalue à retournervalue à la ligne 'si (! Retourvalue && error_file_not_found == marshal.getlastwin32Error ())'.
Cet article de KB est tout ce que vous liez de nombreuses erreurs (où le code source de la réponse sélectionné est venu de) et j'ai perdu ~ 2 jours d'essayer de le faire fonctionner dans tous les paramètres nécessaires. Il est copié collé sur Internet et il existe de nombreux bugs rapportés en fonction de la version du système d'exploitation et de la version IE. P>
Fiddler a été initialement écrit par un employé de Microsoft et est alimenté par FiddlerCore.dll. Telerik (les propriétaires actuels / responsables / vendeurs) de Fiddler mettent toujours à jour, maintiennent et abandonnent la FiddlerCore gratuitement. Si vous ne voulez pas ajouter une référence à FiddlerCore, vous pouvez démonter la DLL et indique la bonne façon d'appeler toutes ces fonctions Wininet horriblement documentées, mais je pense que l'affichage ici serait un mauvais service à Telerik / Plagarisme. p>
Actuellement, FiddlerCore est hébergé ici: http://www.telerik.com/fiddler/fiddlercore a> p>
Le code d'origine de https://support.microsoft.com/en-us/kb/326201 Semble Buggy
Vérification de la documentation MSDN et aussi la version VB ici: https://support.microsoft.com/en-us/kb/262110 P>
J'ai modifié le code comme celui-ci et maintenant pour que cela fonctionne pour moi (le problème était sur l'exécution de FINTNEXTURLCACHEGRoup et FINTNEXTURLCacheenry): P>
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); } } }