Django学习笔记(8)--- Signals和GenericForeignKey的使用

隐身守侯 提交于 2020-02-29 22:21:34

Signals 顾名思义,就是信号的意思。

Django的signals可以用来干什么呢?比如,论坛中别人给你发了一条消息,自动产生一个消息对象。

我们先来自定义一个信号

Message应用中的models.py


from django.db import models
from django.contrib.auth.models import User
from django.dispatch import Signal

notice_signal = Signal(providing_args=['reciver']) #自定义信号

class Message(models.Model):
    sender = models.ForeignKey(User)
    content = models.CharField(max_length=50)

    def newMessage(self,reciver):
        notice_signel.send(sender=self.__class__,reciver=reciver) #发出信号
Notice应用中的models.py



from django.db import models
from django.contrib.auth.modesl import User
from Message.models import notice_signal

class Notice(models.Model):
    reciver = models.ForeignKey(User)

    def sendNotice(sender,reciver,**kwargs):
        Notice.create(reciver=reciver)

notice_signal.connect(sendNotice,sender=Message) #接收信号


这样,只有在创建一个Message对象后,调用newMessage函数,就会自动创建一个Notice对象,实现类似新消息通知的功能。


更复杂的一个例子,我说我要学习,学什么呢,学数据库(一个model),算是学习,学一门语言(另外一个model)也算是学习。我希望每学一样东西,会自动产生一个学习事件model。这样我在查询我所学的知识时,只需要对学习事件进行查询就能够方便获取所有学过的东西的对象。

models.py

from django.db import models
from django.contrib.auth.modesl import User
from django.contrib.contenttypes.models import ContentType
from django.contrib.contenttypes import generic

Class StudyEvent(models.Model):
    #以下三个成员必须指定
    content_type = models.ForeignKey(ContentType)
    object_id = models.PositiveIntegerField()
    content_object = generic.GenericForeignKey('content_type', 'object_id')
    #其它需要字段
    user = models.ForeignKey(User)
    start_time = models.models.DateTimeField(auto_now=True)
    #……

class StudyDatabase(models.Model):
    content = models.CharField(max_length=50)

class StudyLanguage(models.Model):
    content = models.CharField(max_length=50)
因为有时候学数据库和学一门语言要包含一些不同字段,所以需要两个models。


然后,让我们来添加Signal

models.py


#头部添加
from django.db.models.signals import post_save #即对象在创建时会发出的信号

#尾部
def autoCreateStudyEvent(sender,**kwagrs):
    instance = kwargs.get('instance')
    if instance.__class__.__name__ == 'StudyDatabase':
        studyevent = StudyEvent.create(content_object=instance)
    elif instance.__class__.__name__ == 'StudyLanguage':
        studyevent = StudyEvent.create(content_object=instance) 

post_save.connect(autoCreateStudyEvent,sender=StudyDatabase)
post_save.connect(autoCreateStudyEvent,sender=StudyLanguage)

这样,在创建StudyDatebase和StudyLanguage时会自动创建相应的StudyEvent对象。

现在,例如有一个用户叫user1,获取它所有学过的东西只需要:

import datetime

user1 = User.objects.get(username="user1")

studyevents = user1.studyevent_set.filter(start_time=datetime.datetime(2013,8,11)) #获取在8月11日学到的知识

for studyevent in studyevents:

    print studyevent.content_object

这样的好处在于,以后添加了新的models,在获取所有所学过的知识时,不需要重新添加很多代码。


好久没更新博客了,深夜更新暂时写这么多,有错漏请指正。

转载请注明出处,谢谢!

易学教程内所有资源均来自网络或用户发布的内容,如有违反法律规定的内容欢迎反馈
该文章没有解决你所遇到的问题?点击提问,说说你的问题,让更多的人一起探讨吧!