总所周知,Django在1.3版本以后,不能实现: 在模型中删除File或是ImageField中的条目的同时也删除对应服务器上面的文件。这样做是有很多方面的好处和考虑;主要有两点考虑:1: 是防止重要的数据在模型改变的时候丢失(在没有备份的情况下,只改变model数据导致服务器数据丢失),2: 目前考虑到瓶颈不在于服务器的容量上,尽可能的保留server上的文件,目前云端服务器的容量造价低廉,大多数情况下开发中不予考虑
那么我们开始:
-
先在你的Pycharm或是其他IDE中创建一个名字为storage.py的文件,当然你可以随便命名。如下:
#任意文件夹下的 storage.py
from django.core.files.storage import FileSystemStorage
from django.conf import settings
import os
class OverwriteStorage(FileSystemStorage):
def get_available_name(self, name, max_length=None):
if self.exists(name):
os.remove(os.path.join(settings.MEDIA_ROOT, name))
return name
这样做的目的是为为了重写父类FileSystemStorage中的get_available_name方法,从而改变保存后的文件的命名方式。当然你必须再model中调用这个方法,而不是调用父类的get_available_name这个方法:
# models.py 中
class Attribute(models.Model):
member = models.ForeignKey(Member, on_delete=models.CASCADE, null=True)
name = models.CharField(primary_key=True, max_length=30, default='name')
head_img = models.ImageField(storage=OverwriteStorage(), upload_to='headImg')#调用OverwriteStorage()方法
last_login_time = models.DateTimeField('last login time', default=timezone.now())
这样的目的是:如果模型中已经存在了相同的文件名或是图片名(我们的代码演示的是图片名),那么再次保存相同的文件或是图片名称的时候就会去先核对有没有相同命名的文件或是图片,如果有的话就删掉,然后再保存。这样就等同于"覆盖"。
这种模型主要用来处理客户的头像和一次性数据,如果客户修改了头像那么原有的图像就没有保留的意义。
下面解决第二个问题(从Django中提供的admin中删除模型,没有办法同时删掉与其想关联的文件或是图片的解决办法;
- 方案一 : 在models.py 中添加receiver来确保如果你删除model中的条目,那么与其想关联的服务器文件也一并被删除,代码如下:
@receiver(post_delete, sender=Attribute)
def mymodel_delete(sender,instance, ** kwargs):
instance.head_img.delete(False)# False 表示model不保存
-
方案二
如果你嫌麻烦,或是没有感觉到我说的1.3版本以后那两个改变的优点对于你非常有用,或是你认为那只是Django的一坨shit,那么你直接利用一个叫做django-cleanup的软件来到达Django1.3之前的样子,具体请参考:
django-cleanup