2

我正在尝试使模型上的视图级缓存过期post_save(通过https://docs.djangoproject.com/en/1.3/topics/cache/?from=olddocs#the-per-view-cache设置) . 我做了一些谷歌搜索,并在 SO:Expire a view-cache in Django 上找到了这个答案:Expire a view-cache in Django? 但这对我不起作用。

我在 freenode 上的#django 房间四处询问,一致认为这可能是由于最近在 1.3 中所做的缓存更改

有谁知道我如何清除一个模型的缓存条目get_absolute_url()

4

2 回答 2

4

我想到了!

ilvar为我指明正确的方向而欢呼。我的实现如下。我创建了一个名为的属性cache_key,并将 post_save 接收器添加到模型的子类中,这些模型的视图级缓存在更新后需要清除。欢迎提出改进建议!

from django.conf import settings
from django.core.cache import cache
from django.db.models.signals import post_save
from django.http import HttpRequest
from django.utils.cache import _generate_cache_header_key

from someapp.models import BaseModelofThisClass

class SomeModel(BaseModelofThisClass):
    ...
    @property
    def cache_key(self):
        # Create a fake request object
        request = HttpRequest()
        # Set the request method
        request.method = "GET"
        # Set the request path to be this object's permalink.
        request.path = self.get_absolute_url()
        # Grab the key prefix (if it exists) from settings
        key_prefix = settings.CACHE_MIDDLEWARE_KEY_PREFIX
        # Generate this object's cache key using django's key generator
        key = _generate_cache_header_key(key_prefix, request)
        # This is a bit hacky but necessary as I don't know how to do it
        # properly otherwise. While generating a cache header key, django
        # uses the language of the HttpRequest() object as part of the
        # string. The fake request object, by default, uses
        # settings.LANGUAGE_CODE as it's language (in my case, 'en-us')
        # while the true request objects that are used in building views
        # use settings.LANGUAGES ('en'). Instead of replacing the segment
        # of the string after the fact it would be far better create a more
        # closely mirrored HttpRequest() object prior to passing it to
        # _generate_cache_header_key().
        key = key.replace(settings.LANGUAGE_CODE, settings.LANGUAGES[settings.DEFAULT_LANGUAGE][0])

        return key

    @receiver(post_save)
    def clear_cache_for_this_item(sender, instance, **kwargs):
        # If this is a sub-class of another model
        if sender not in BaseModelofThisClass.__subclasses__():
            return
        else:
            cache.delete(instance.cache_key)
于 2012-05-07T14:59:33.523 回答
3

Django 的缓存中间件使用为请求生成缓存键。因此,您可以request使用所需路径进行伪造并获取缓存键。然后只需将其从缓存中删除。

PScache_page装饰器也使用该中间件,因此它应该以相同的方式工作。

于 2012-05-05T04:33:10.903 回答