Django筆記 —— 模型高級進階


  最近在學習Django,打算玩玩網頁后台方面的東西,因為一直很好奇但卻沒怎么接觸過。Django對我來說是一個全新的內容,思路想來也是全新的,或許並不能寫得很明白,所以大家就湊合着看吧~

  本篇筆記(其實我的所有筆記都是),並不會過於詳細的講解。因此如果有大家看不明白的地方,歡迎在我正版博客下留言,有時間的時候我很願意來這里與大家探討問題。(當然,不能是簡簡單單就可以百度到的問題-.-)

  我所選用的教材是《The Django Book 2.0》,本節是第十章,模型高級進階。


  在基礎部分的學習中,我體會到了一點經驗:傻瓜教程最適合作為本書的筆記了~因為本書對於原理講得很細,看一遍也就能基本理解,但由於講得太細,具體操作步驟正是其不足。因此,讀這本書,如果配上操作教程式的筆記,那復習起來就很舒服了 ^.^

  因此,高級部分的筆記,將給出很多操作教程,沒看過書的同學請先看了書再來看筆記~


 

0. 目錄

  1. 模型回顧與初探

    1.1 訪問外鍵(Foreign Key)值

    1.2 訪問多對多(Many-to-Many)值

  2. 更改數據庫模式(Database Schema)

  3. Managers

    3.1 增加額外Manager的方法

    3.2 修改初始Manager QuerySets

  4. 模型方法

  5. 執行原始SQL查詢

1. 模型回顧與初探

  模型代碼如下:

from django.db import models

class Publisher(models.Model):
    name = models.CharField(max_length=30)
    address = models.CharField(max_length=50)
    city = models.CharField(max_length=60)
    state_province = models.CharField(max_length=30)
    country = models.CharField(max_length=50)
    website = models.URLField()

    def __unicode__(self):
        return self.name

class Author(models.Model):
    first_name = models.CharField(max_length=30)
    last_name = models.CharField(max_length=40)
    email = models.EmailField()

    def __unicode__(self):
        return u'%s %s' % (self.first_name, self.last_name)

class Book(models.Model):
    title = models.CharField(max_length=100)
    authors = models.ManyToManyField(Author)
    publisher = models.ForeignKey(Publisher)
    publication_date = models.DateField()

    def __unicode__(self):
        return self.title

  1.1 訪問外鍵(Foreign Key)值

    Book的外鍵是Publisher,首先,訪問Book代碼如下:

# Book

>>> from mysite.books.models import Book
>>> b = Book.objects.get(id=50)
>>> b.title
u'The Django Book'

     然后,通過Book訪問Publisher代碼如下:

# Publisher

>>> b = Book.objects.get(id=50)
>>> b.publisher
<Publisher: Apress Publishing>
>>> b.publisher.website
u'http://www.apress.com/'

     通過Publisher訪問Book的代碼則如下:

# Book1

>>> p = Publisher.objects.get(name='Apress Publishing')
>>> p.book_set.all()
[<Book: The Django Book>, <Book: Dive Into Python>, ...]

# Book2

>>> p = Publisher.objects.get(name='Apress Publishing')
>>> p.book_set.filter(name__icontains='django')
[<Book: The Django Book>, <Book: Pro Django>]

   1.2 訪問多對多(Many-to-Many)值

    通過Book訪問Author代碼如下:

# Author

>>> b = Book.objects.get(id=50)
>>> b.authors.all()
[<Author: Adrian Holovaty>, <Author: Jacob Kaplan-Moss>]
>>> b.authors.filter(first_name='Adrian')
[<Author: Adrian Holovaty>]
>>> b.authors.filter(first_name='Adam')
[]

     通過Author訪問Book代碼則如下:

# Book

>>> a = Author.objects.get(first_name='Adrian', last_name='Holovaty')
>>> a.book_set.all()
[<Book: The Django Book>, <Book: Adrian's Other Book>]

  

2. 更改數據庫模式(Database Schema)

  當你修改模型時(例如添加字段、刪除字段、刪除模型……),應當依次進行以下步驟:

    (1) 修改models.py中模型

    (1+) 這時你的admin頁面依舊正常運行,查看時也的確是修改完畢的樣子,但是一旦添加(修改)對象則會報錯

    (2) 生成migration  python manage.py makemigrations appname 

    (3) 激活模型(migrate)  python manage.py migrate 

    (3+) 現在你可以正常使用了 ^.^

  上述過程肯定是沒問題的,除非你修改的是外鍵,那么就會很麻煩了。

  例如,你刪除了Book模型的外鍵,再要恢復則會在第2步遇到這樣的提示:

    You are trying to add a non-nullable field 'publisher' to book without a default; we can't do that (the database needs something to populate existing rows).
    Please select a fix:
    1) Provide a one-off default now (will be set on all existing rows)
    2) Quit, and let me add a default in models.py
    Select an option:

  我不知道應該如何給值,於是選的2,結果……再也無法打開Book了……報錯信息如下:

    OperationalError at /admin/books/book/

    (1054, "Unknown column 'books_book.publisher_id' in 'field list'")

  因此,在我知道如何做之前,權宜之計,我不會嘗試修改外鍵!

 

3. Managers

  我們之前一直在使用的Book.objects.all()之類的語句,其Book.objects就是所謂的Manager,這是Django定義的用來管理模型的類,其中定義了很多函數,例如.all()。

  而其中各種函數,例如all()、get(),返回的有類似列表的QuerySet,也有單個對象。

  3.1 增加額外Manager的方法

    有時候objects的功能不夠用,我們就得寫自己的Manager,代碼如下:

 1 # models.py
 2 
 3 from django.db import models
 4 
 5 # ... Author and Publisher models here ...
 6 
 7 class BookManager(models.Manager):
 8     def title_count(self, keyword):
 9         return self.filter(title__icontains=keyword).count()
10 
11 class Book(models.Model):
12     title = models.CharField(max_length=100)
13     authors = models.ManyToManyField(Author)
14     publisher = models.ForeignKey(Publisher)
15     publication_date = models.DateField()
16     num_pages = models.IntegerField(blank=True, null=True)
17     objects = BookManager()
18 
19     def __unicode__(self):
20         return self.title

    上述代碼的第7~9行定義了自己的Manager,而第17行則把objects設為了這個Manager。

    如此一來,我們便可以做如下操作了:

>>> Book.objects.title_count('django')
4
>>> Book.objects.title_count('python')
18

    如果我們不定義objects,那么Django會自動生成;而現在咱們重新定義了objects,那么它就是咱們定義的這個Manager了。

    多說一句,上面代碼的作用,是查找所有書籍中名字帶有指定字符串的書籍數量。

  3.2 修改初始Manager QuerySets

    我們在寫出Book.objects()的時候,Django會認為我們選中了所有的Book對象,這是因為默認的objects類中有一個函數:get_query_set(),這個函數決定了你調用objects的時候選中哪些對象,它默認是返回模型中所有對象的。

    那么,我們便可以改寫一下這個函數,來讓我們選中的對象(Query Set)改變一下,代碼如下:

from django.db import models

# First, define the Manager subclass.
class DahlBookManager(models.Manager):
    def get_query_set(self):
        return super(DahlBookManager, self).get_query_set().filter(author='Roald Dahl')

# Then hook it into the Book model explicitly.
class Book(models.Model):
    title = models.CharField(max_length=100)
    author = models.CharField(max_length=50)
    # ...

    objects = models.Manager() # The default manager.
    dahl_objects = DahlBookManager() # The Dahl-specific manager.

    如此一來,下面的Book.dahl_objects.all()便是上面函數所選的對象了:作者是Roald Dahol的書。

Book.dahl_objects.all()
Book.dahl_objects.filter(title='Matilda')
Book.dahl_objects.count()

    再比如,我們可以在一個模型中實現幾個不同的Manager,代碼如下:

class MaleManager(models.Manager):
    def get_query_set(self):
        return super(MaleManager, self).get_query_set().filter(sex='M')

class FemaleManager(models.Manager):
    def get_query_set(self):
        return super(FemaleManager, self).get_query_set().filter(sex='F')

class Person(models.Model):
    first_name = models.CharField(max_length=50)
    last_name = models.CharField(max_length=50)
    sex = models.CharField(max_length=1, choices=(('M', 'Male'), ('F', 'Female')))
    people = models.Manager()
    men = MaleManager()
    women = FemaleManager()

 

4. 模型方法

  我們之前一直在使用模型自帶的方法,現在介紹如何自己定義方法:

from django.contrib.localflavor.us.models import USStateField
from django.db import models

class Person(models.Model):
    first_name = models.CharField(max_length=50)
    last_name = models.CharField(max_length=50)
    birth_date = models.DateField()
    address = models.CharField(max_length=100)
    city = models.CharField(max_length=50)
    state = USStateField() # Yes, this is U.S.-centric...

    def baby_boomer_status(self):
        "Returns the person's baby-boomer status."
        import datetime
        if datetime.date(1945, 8, 1) <= self.birth_date <= datetime.date(1964, 12, 31):
            return "Baby boomer"
        if self.birth_date < datetime.date(1945, 8, 1):
            return "Pre-boomer"
        return "Post-boomer"

    def is_midwestern(self):
        "Returns True if this person is from the Midwest."
        return self.state in ('IL', 'WI', 'MI', 'IN', 'OH', 'IA', 'MO')

    def _get_full_name(self):
        "Returns the person's full name."
        return u'%s %s' % (self.first_name, self.last_name)
    full_name = property(_get_full_name)

  之后,使用代碼如下:

>>> p = Person.objects.get(first_name='Barack', last_name='Obama')
>>> p.birth_date
datetime.date(1961, 8, 4)
>>> p.baby_boomer_status()
'Baby boomer'
>>> p.is_midwestern()
True
>>> p.full_name  # Note this isn't a method -- it's treated as an attribute
u'Barack Obama'

 

5. 執行原始SQL查詢

  最后,有時候會發現Django定義的內容不全,我們需要寫一些sql代碼,這時候就要:

>>> from django.db import connection
>>> cursor = connection.cursor()
>>> cursor.execute("""
...    SELECT DISTINCT first_name
...    FROM people_person
...    WHERE last_name = %s""", ['Lennon'])
>>> row = cursor.fetchone()
>>> print row
['John']

  實際寫入代碼中的時候,最好寫在自己的Manager中,就像這樣:

from django.db import connection, models

class PersonManager(models.Manager):
    def first_names(self, last_name):
        cursor = connection.cursor()
        cursor.execute("""
            SELECT DISTINCT first_name
            FROM people_person
            WHERE last_name = %s""", [last_name])
        return [row[0] for row in cursor.fetchone()]

class Person(models.Model):
    first_name = models.CharField(max_length=50)
    last_name = models.CharField(max_length=50)
    objects = PersonManager()

  如此,便可這樣使用:

>>> Person.objects.first_names('Lennon')
['John', 'Cynthia']

 


  至此,“模板高級進階”內容完結,下一篇是——“通用視圖”。


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM