zoukankan      html  css  js  c++  java
  • 关于django Models的个人理解和related_name的使用

      作为一个新人(刚刚大学还没有毕业就出来实习,可以说是真的什么都不知到,什么都要重新学,但是这样真的可以锻炼自己的意志力和能力)。现在在公
    司是前端和后端一起坐,所以要学的东西是真的多的让人想不到。在学习的过程中也越到了不少的问题,今天和大家分享一下django中的models的一些知识,
    有不对或者不当的地方希望大家可以指正,大家共同学习,共同进步,在此谢过!
      相信做程序的大家都是知道的,关系数据库的强大不仅仅是由于用数据库可以存放数据,更重要的是他可以存放数据库中表与表之间的关系,Django请
    提供了三种最为通用的数据库关系模型
       1 mang-to-many(对对多关系)
       2 many-to-one(多对一关系)
      3 one-to-one(一对一关系) 
     
    一,多对一关系:
      用 ForeignKey 来定义多对一的关系。用法和其他的 Field 是一样的,把它放到模型中类的属性定义中就行了。ForeignKey 需要一个与之相关联的类作为

    位置参数。在这里我用django book中的一个比较经典的实例来和大家一起分享一下:首先,定义了三个模型分别为:Publisher(出版商),Author(作者)和Book(书)

     1 class Publisher(models.Model):
     2     name = models.CharField(max_length=30)
     3     address = models.CharField(max_length=50)
     4     city = models.CharField(max_length=60)
     5     state_province = models.CharField(max_length=30)
     6     country = models.CharField(max_length=50)
     7     website = models.URLField()
     8     def __unicode__(self):
     9          return self.name
    10 
    11 
    12 class Meta:
    13         ordering = ['name']
    14 
    15 
    16 class Author(models.Model):
    17     first_name = models.CharField(max_length=30)
    18     last_name = models.CharField(max_length=40)
    19     email = models.EmailField(blank=True)
    20     def __unicode__(self):
    21          return  u'%s %s' % (self.first_name, self.last_name)
    22 
    23 
    24 class Book(models.Model):
    25     title = models.CharField(max_length=100)
    26     authors = models.ManyToManyField(Author)
    27     publisher = models.ForeignKey(Publisher, related_name = "publisher_set")
    28     publication_date = models.DateField(blank=True, verbose_name='e-mail')
    29     num_pages = models.IntegerField(blank=True, null=True)
    30     # objects = BookManager()
    31     objects = models.Manager()           # The default manager.
    32     dahl_objects = DahlBookManager()     # The Dahl-specific manager
    33 
    34     # .
    35     def __unicode__(self):
    36          return self.title

    要建立一个递归的关系,即一个对象和自身的多对一关系,你可以这样写:models.ForeignKey('self') :

     1 class Employee(models.Model): 2 manager = models.ForeignKey('self') 

    如果你创建关系时,所需的模型还没有被定义,你可以不使用模型对象本身,而是使用那个模型的名字。

    1 class Car(models.Model):
    2 manufacturer = models.ForeignKey('Manufacturer')
    3 ...
    4 class Manufacturer(models.Model):

    但是,你要记住,只能对在同一个 models.py 文件中的模型使用字符串引用,对于其他应用程序中的模型或者从其他地方导入
    的模型是不能使用名字对其做引用的。
    Django在数据库中使用的列名称是对应的字段的名称后追加 _id 得到的字符串。再前面的那个例子中, Car 模型对应的数据库
    表中会有一个名字是 manufacturer_id 的列,(你可以通过指定 db_column 来显式改变这个名字,参见前面
    的db_column一节)但是,如果你不需要写定制的SQL语句的话,你永远不要去处理数据库列名,只需要处理你的模型对象中的
    字段名称。

    在这里有几个概念要讲清楚,当初我就是载在这上面的,希望有同感的朋友可以看到,ForeignKey选项

    1.edit_inline:

    如果不设为 False 的话,它对应的对象就可以在页面上内联编辑,就是说这个对象有自 己独立的
    管理界面。如果设为 models.TABULAR 或者 models.STACKED 的话,这个内 联编辑对象分别显
    示成一个表格或者一些字段的集合。

    2.limit_choices_to:

    可以限定对象的值的范围的一个参数和值的字典。结合Python的 datetime 模块的

    函数可以根据日期来限定对象。例如,下面的代码:
    limit_choices_to = {'pub_date__lte': datetime.now}
    把可选对象限定到 pub_date 早于当前时间的对象中。
    除字典外,这里也可以是一个可以执行更复杂的查询的 Q 对象
    这个选项和 edit_inline 是不兼容的。

    3.max_num_in_admin

    于内联编辑对象,这个是要在管理界面里显示的相关对象的最多个数。所以,如果披萨最多 只会
    有10种配料, max_num_in_admin=10 会保证用户最多输入10种配料。
    记住,本项并不保证不会创建10种以上的配料,他只是控制管理界面,而不是在Python的API 层和
    数据库层做什么限制。

    4.min_num_in_admin

    在管理界面中要显示的相关的对象的最少个数。通常,在创建的时候,显示的内联对象的个数 为
    num_in_admin 个,在编辑的时候,在当前的基础上又会多显示 num_extra_on_change 个空对
    象,但是显示的对象个数不会少于 min_num_in_admin 个。

    5.num_extra_on_change

    修改对象时要额外显示的对象数目。

    6.num_in_admin

    添加对象时要显示的内联对象的默认个数。

    7.raw_id_admin

    为要键入的整数显示一个文本框,而不是一个下拉列表。在关联对象有很多行时,这个比显示 一个
    列表选择框更实用。
    使用 edit_inline 时,本项无效。

    8.related_name

    关联对象反向引用描述符。

    9.to_field

    关联对象的用于关联的字段,Django默认使用关联对象的主键。

    要学习ForeignKey并了解深刻的话,可以先对以上ForeignKey的这些选项有一定的了解。这样可以加深你对ForeignKey的印象。

      万事俱备,现在,假如你想通过Book这张表去访问Publisher和Author都是可以直接访问的,因为Book这张表中有外键和这两张

    表关联,因此可以直接通过语句去访问Book和Publisher表。

    1 def get_info(request, id):
    2     books = Book.objects.filter(publisher__id = id).all()
    3     return render_to_response("showdb_info.html",{
    4         "books": books
    5     },
    6      context_instance=RequestContext(request))

    这一步相信大家都是知道的,即使不知道,等你看了笔者的这篇博客之后肯定也会一目了然的。但是,假如反过来该怎么去查询呢?

    (通过Publisher或者Author表查询Book表中的内容)在这里就要用到一个相当重要的ForeignKey选项,通过该选项你可以随便访问Books

    中的任意字段。

    def get_book_info(request, id):
        publisher = Publisher.objects.filter(publisher_set__id = id)
        return render_to_response("showdb_info.html",{
            "publisher": publisher,
        },
         context_instance=RequestContext(request))

    以上python函数中的都是伪代码,有兴趣的读者可以自己设计自己的代码。此文将会持续更新,有意者可以持续关注,欢迎大家指正不足!

     
     
  • 相关阅读:
    全局对象
    公用属性与原型链
    内存图
    JS中数据类型的转换
    for in 遍历
    JS里的数据类型
    Collect~Something else
    Spring Batch学习笔记(一)
    C#之WinForm设置控件居中
    C#之使用AutoUpdater自动更新客户端
  • 原文地址:https://www.cnblogs.com/blogofwyl/p/4283513.html
Copyright © 2011-2022 走看看