关于Django外键赋值问题详解


Posted in Python onAugust 13, 2017

本文主要给大家介绍关于Django外键赋值的相关内容,分享出来供大家参考学习,在开始之前,我们先来看一段代码:

class Article(models.Model):
 title = models.CharField(max_length=1024, default='')
 ...
 def __str__(self):
  return 'Article pk:%d %s' % (self.pk, self.title[:30])

class ArticleContent(models.Model):
 article = cached_fields.OneToOneField(Article)
 ...

写代码的的时候,发现了一个很奇怪的现象,当我给一个instance的外键(以_id结尾)赋值(数字)的时候 ,这个外键对应的instance的值并不会改变。

In [44]: ac = ArticleContent.objects.get(article_id=14269)
In [45]: ac.article_id
Out[45]: 14269
In [46]: ac.article_id = 14266
In [47]: ac.save()
In [48]: ac.article
Out[48]: <Article: Article pk:14266 EC: Russia, Ukraine to Meet in>
In [49]: ac.article.pk
Out[49]: 14266

如上面的代码所示,为了找到答案,我翻了一下Django的源码:

django/db/models/fields/related_descriptors.py 
  def __get__(self, instance, cls=None):
   """
   Get the related instance through the forward relation.

   With the example above, when getting ``child.parent``:

   - ``self`` is the descriptor managing the ``parent`` attribute
   - ``instance`` is the ``child`` instance
   - ``cls`` is the ``Child`` class (we don't need it)
   """
   if instance is None:
    return self

   # The related instance is loaded from the database and then cached in
   # the attribute defined in self.cache_name. It can also be pre-cached
   # by the reverse accessor (ReverseOneToOneDescriptor).
   try:
    rel_obj = getattr(instance, self.cache_name)
   except AttributeError:
    val = self.field.get_local_related_value(instance)
    if None in val:
     rel_obj = None
    else:
     qs = self.get_queryset(instance=instance)
     qs = qs.filter(self.field.get_reverse_related_filter(instance))
     # Assuming the database enforces foreign keys, this won't fail.
     rel_obj = qs.get()
     # If this is a one-to-one relation, set the reverse accessor
     # cache on the related object to the current instance to avoid
     # an extra SQL query if it's accessed later on.
     if not self.field.remote_field.multiple:
      setattr(rel_obj, self.field.remote_field.get_cache_name(), instance)
    setattr(instance, self.cache_name, rel_obj)

   if rel_obj is None and not self.field.null:
    raise self.RelatedObjectDoesNotExist(
     "%s has no %s." % (self.field.model.__name__, self.field.name)
    )
   else:
    return rel_obj

注释得非常到位,当我们请求ac.article的时候,会先去检查对应的cache(在这里是_article_cache,感兴趣可以去看cache_name的生成规则,就是外键名前面加下划线,后面加cache)存不存在,如果不存在那么就进行数据库请求,请求完之后会保存到cache中。

我们再看看__set__ ,代码太长就不贴了(就在__get__方法下面)。除了给外键字段(article)赋值外,还会将pk字段(article_id,是lh_field.attname的值)设置为None,这样下次请求的时候就能拿到正确的值。

以上都是ForeignKey的Magic,而当我们给article_id赋值的时候,只是在给一个普通的attribute赋值而已,没有任何magic,不会清理对应外键的cache,这时候拿到的instance仍然是cache中原来的那个instance。

总结

以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作能带来一定的帮助,如果有疑问大家可以留言交流,谢谢大家对三水点靠木的支持。

Python 相关文章推荐
利用Python的Flask框架来构建一个简单的数字商品支付解决方案
Mar 31 Python
Python操作MySQL数据库9个实用实例
Dec 11 Python
Python 爬虫模拟登陆知乎
Sep 23 Python
Python 数据结构之旋转链表
Feb 25 Python
获取python文件扩展名和文件名方法
Feb 02 Python
python如何制作英文字典
Jun 25 Python
python 如何去除字符串头尾的多余符号
Nov 19 Python
Pytorch中实现只导入部分模型参数的方式
Jan 02 Python
Python对Tornado请求与响应的数据处理
Feb 12 Python
部署Django到阿里云服务器教程示例
Jun 03 Python
pycharm不以pytest方式运行,想要切换回普通模式运行的操作
Sep 01 Python
Python办公自动化解决world文件批量转换
Sep 15 Python
python爬虫实战之最简单的网页爬虫教程
Aug 13 #Python
详解python中executemany和序列的使用方法
Aug 12 #Python
mysql 之通过配置文件链接数据库
Aug 12 #Python
python+selenium开发环境搭建图文教程
Aug 11 #Python
Python实现的递归神经网络简单示例
Aug 11 #Python
Python调用系统底层API播放wav文件的方法
Aug 11 #Python
Django 导出 Excel 代码的实例详解
Aug 11 #Python
You might like
PHP的cURL库功能简介 抓取网页、POST数据及其他
2011/04/07 PHP
PHP处理bmp格式图片的方法分析
2017/07/04 PHP
PHP语言对接抖音快手小红书视频/图片去水印API接口源码
2020/08/11 PHP
jQuery Tab插件 用于在Tab中显示iframe,附源码和详细说明
2011/06/27 Javascript
基于jQuery替换table中的内容并显示进度条的代码
2011/08/02 Javascript
javascript笔记 String类replace函数的一些事
2011/09/22 Javascript
Javascript中匿名函数的多种调用方式总结
2013/12/06 Javascript
JS合并数组的几种方法及优劣比较
2014/09/19 Javascript
jQuery Ajax()方法使用指南
2014/11/19 Javascript
JavaScript严格模式详解
2015/11/18 Javascript
javascript时间差插件分享
2016/07/18 Javascript
运用js教你轻松制作html音乐播放器
2020/04/17 Javascript
AngularJS extend用法详解及实例代码
2016/11/15 Javascript
基于Marquee.js插件实现的跑马灯效果示例
2017/01/25 Javascript
推荐三款日期选择插件(My97DatePicker、jquery.datepicker、Mobiscroll)
2017/04/21 jQuery
在vue项目中引入highcharts图表的方法(详解)
2018/03/05 Javascript
js实现移动端吸顶效果
2020/01/08 Javascript
js实现简单的秒表
2020/01/16 Javascript
JS面向对象编程基础篇(三) 继承操作实例详解
2020/03/03 Javascript
Vue实现附件上传功能
2020/05/28 Javascript
解决Vue中使用keepAlive不缓存问题
2020/08/04 Javascript
解决vue项目中出现Invalid Host header的问题
2020/11/17 Javascript
Python框架Flask的基本数据库操作方法分析
2018/07/13 Python
对Python 语音识别框架详解
2018/12/24 Python
python获取地震信息 微信实时推送
2019/06/18 Python
简单了解python的break、continue、pass
2019/07/08 Python
python 串口读取+存储+输出处理实例
2019/12/26 Python
css3+伪元素实现鼠标移入时下划线向两边展开的效果
2017/04/25 HTML / CSS
AmazeUI的JS表单验证框架实战示例分享
2020/08/21 HTML / CSS
Omio中国:全欧洲低价大巴、火车和航班搜索和比价
2018/08/09 全球购物
廉洁自律承诺书
2014/03/27 职场文书
世界读书日的活动方案
2014/08/20 职场文书
私人贷款担保书该怎么写呢?
2019/07/02 职场文书
python3+PyQt5+Qt Designer实现界面可视化
2021/06/10 Python
springboot实现string转json json里面带数组
2022/06/16 Java/Android
阿里云服务器(windows)手动部署FTP站点详细教程
2022/08/05 Servers