好男人天堂网,久久精品国产这里是免费,国产精品成人一区二区,男人天堂网2021,男人的天堂在线观看,丁香六月综合激情

當前位置:首頁 > 網(wǎng)站舊欄目 > 學習園地 > 設(shè)計軟件教程 > 翻譯www.djangobook.com之第五章:與數(shù)據(jù)庫交互:模型

翻譯www.djangobook.com之第五章:與數(shù)據(jù)庫交互:模型
2010-01-13 23:36:18  作者:  來源:
第3章我們談到了用Django構(gòu)建動態(tài)網(wǎng)站,設(shè)置視圖和URL配置
如我們所說,試圖負責邏輯和返回應(yīng)答,例子中我們計算了當前的日期和時間
現(xiàn)在的Web程序中常常和數(shù)據(jù)庫打交道
一個數(shù)據(jù)庫驅(qū)動的網(wǎng)站在后臺連接數(shù)據(jù)庫服務(wù)器,得到并顯示很好的格式化的Web頁面
同樣,網(wǎng)站也可以提供給訪問者也具有操作數(shù)據(jù)庫的功能
許多復(fù)雜的網(wǎng)站以上兩種功能的結(jié)合,如Amazon.com就是一個數(shù)據(jù)庫驅(qū)動的站點
每一個產(chǎn)品頁面都是Amazon數(shù)據(jù)庫格式后的HTML,你訪問頁面也就是間接訪問數(shù)據(jù)庫
Django很適合數(shù)據(jù)庫驅(qū)動的網(wǎng)站,通過Python它提供強大的數(shù)據(jù)庫訪問能力
這章將講述Django的數(shù)據(jù)庫層

在視圖里進行數(shù)據(jù)庫查詢的“啞”方式
前一章講到通過在視圖里硬編碼HTML來輸出HTML的“啞”方式,在視圖里也有得到數(shù)據(jù)庫數(shù)據(jù)的“啞”方式
這很簡單,只是使用一些Python庫執(zhí)行SQL查詢并且處理結(jié)果
在下面的例子里我們使用MySQLdb庫(可以在如下地址得到http://sourceforge.net/projects/mysql-python)
來連接MySQL數(shù)據(jù)庫,得到一些記錄來填充模板,并顯示到Web頁面上:
Java代碼 復(fù)制代碼
  1. from django.shortcuts import render_to_response   
  2. import MySQLdb   
  3.   
  4. def book_list(request):   
  5.     db = MySQLdb.connect(user='me', db='mydb', passwd='secret', host='localhost')   
  6.     cursor = db.cursor()   
  7.     cursor.execute('SELECT name FROM books ORDER BY name')   
  8.     names = [row[0for row in cursor.fetchall()]   
  9.     db.close()   
  10.     return render_to_response('book_list.html', {'names': names})  

這個方法可以工作,但是馬上一些問題出來了:
1,我們把數(shù)據(jù)庫連接的參數(shù)硬編碼到代碼里面了,理想狀況下它們應(yīng)該存儲在Django配置里面
2,我們必須寫一些樣板文件代碼,如建立連接,創(chuàng)建cursor,執(zhí)行語句和關(guān)閉連接等
細想狀況下,我們應(yīng)該只需指出我們需要什么結(jié)果
3,它把我們和MySQL綁在一起,如果我們想切換到PostgreSQL
我們必須使用不同的數(shù)據(jù)庫適配器psycopg,改變數(shù)據(jù)庫連接參數(shù)以及考慮重寫SQL語句
理想狀況下數(shù)據(jù)庫服務(wù)器應(yīng)該是抽象的,替換數(shù)據(jù)庫應(yīng)該只在一個地方設(shè)置
你可能會想,Django的數(shù)據(jù)庫層的目標應(yīng)該是解決這些問題,下面簡單看看怎樣用Django數(shù)據(jù)庫API重寫上面代碼:
Java代碼 復(fù)制代碼
  1. from django.shortcuts import render_to_response   
  2. from mysite.books.models import Book   
  3.   
  4. def book_list(request):   
  5.     books = Book.objects.order_by('name')   
  6.     return render_to_response('book_list.html', {'books': books})  

我們在本章稍后解釋這些代碼,現(xiàn)在先感覺一下它的樣子

MTV開發(fā)模式
在我們專研更多的代碼之前,讓我們先花點時間考慮一些Django Web程序的整體設(shè)計
前面的章節(jié)我們提到,Django設(shè)計來鼓勵松耦合和分離程序模塊
如果你遵循這個哲學,改變一部分代碼而不影響其它模塊是很容易做到的
例如在視圖方法里,我們討論了使用模板來分離業(yè)務(wù)邏輯和呈現(xiàn)邏輯的重要性
在數(shù)據(jù)庫層的數(shù)據(jù)訪問邏輯我們將遵循同樣的哲學
數(shù)據(jù)訪問,業(yè)務(wù)邏輯和呈現(xiàn)邏輯組成常說的“Model View Controller”(MVC)軟件架構(gòu)模式
“Model”指數(shù)據(jù)訪問層,“View”指系統(tǒng)中選擇什么來呈現(xiàn)以及怎樣呈現(xiàn)的部分
“Controller”則指系統(tǒng)中通過用戶輸入決定使用哪個視圖及訪問必要的模型的部分
采用MVC,MTV等縮寫只是便于開發(fā)人員溝通
Django遵循了MVC模式,它可以被稱位MVC框架,下面是M,V,C在Django中的位置:
1,M,數(shù)據(jù)據(jù)訪問部分,通過Django的數(shù)據(jù)庫層處理,也就是本章所講述的內(nèi)容
2,V,選擇數(shù)據(jù)并決定怎樣呈現(xiàn)的部分,通過視圖和模板來處理
3,C,控制部分通過Django框架本身的URL配置和對Python方法的調(diào)用來處理
因為“C”是Django框架本身處理而導致Django大部分精彩的東西在于模型,模板和視圖
所以Django被稱位MTV框架:
1,M,代表模型,是數(shù)據(jù)訪問層,它包含了關(guān)于數(shù)據(jù)的一切東西,怎樣得到數(shù)據(jù),怎樣驗證數(shù)據(jù),
它具有什么行為以及數(shù)據(jù)之間的關(guān)系
2,T,代表模板,是展現(xiàn)層,它包含了呈現(xiàn)相關(guān)的決策,如內(nèi)容怎樣在Web頁面中顯示以及其它類型的文檔
3,V,代表視圖,是業(yè)務(wù)邏輯層,它包含了訪問模型的邏輯和選擇合適的模板
你可以認為視圖是模型和模板的橋梁
如果你對MVC框架熟悉,如Ruby on Rails,你可以把Django的視圖想象成“controllers”,
把Django的模板想象成“views”,這是對MVC的不同解釋造成的不幸的混亂
在Django關(guān)于MVC的解釋中,“view”描述呈現(xiàn)給用戶的數(shù)據(jù)
沒有必要弄清數(shù)據(jù)怎樣顯示,而是描述哪個數(shù)據(jù)應(yīng)該被呈現(xiàn)
對比而言,Ruby on Rails以及類似的框架建議controller的工作包括決定哪個數(shù)據(jù)顯示給用戶,
視圖嚴格的決定數(shù)據(jù)怎樣顯示,而不是決定哪個數(shù)據(jù)來顯示
每一個解釋都不比另一個正確,最重要的事情是理解底層的概念

配置數(shù)據(jù)庫
所有的哲學牢記在心之后,讓我們開始發(fā)掘Django的數(shù)據(jù)庫層
首先我們注意一些細小的配置,我們需要告訴Django使用哪個數(shù)據(jù)庫和怎樣連接它
我們假設(shè)你已經(jīng)有了一個數(shù)據(jù)庫服務(wù)器,啟動它并創(chuàng)建一個database(使用CREATE DATABASE語句)
SQLite是一個特例,不需要創(chuàng)建database,因為SQLite在文件系統(tǒng)上使用單獨的文件存儲數(shù)據(jù)
和上一章的TEMPLATE_DIRS一樣,數(shù)據(jù)庫配置在Django配置文件里面,默認是settings.py
Java代碼 復(fù)制代碼
  1. DATABASE_ENGINE = ''  
  2. DATABASE_NAME = ''  
  3. DATABASE_USER = ''  
  4. DATABASE_PASSWORD = ''  
  5. DATABASE_HOST = ''  
  6. DATABASE_PORT = ''  

我們來看看每個配置是什么意思:
1,DATABASE_ENGINE告訴Django使用哪個數(shù)據(jù)庫引擎,如果你使用數(shù)據(jù)庫和Django工作的話,
DATABASE_ENGINE必須是下面的字符串集合:
引用

設(shè)置                         數(shù)據(jù)庫               需要的適配器
postgresql             PostgreSQL       psycopg version 1.x, http://initd.org/projects/psycopg1

postgresql_psycopg2    PostgreSQL       psycopg version 2.x, http://initd.org/projects/psycopg2

mysql                  MySQL            MySQLdb, http://sourceforge.net/projects/mysql-python

sqlite3                SQLite No adapter needed if using Python 2.5+ Otherwise, pysqlite, http://initd.org/tracker/pysqlite

ado_mssql        Microsoft SQL Server   adodbapi version 2.0.1+, http://adodbapi.sourceforge.net/

oracle                 Oracle           cx_Oracle, http://www.python.net/crew/atuining/cx_Oracle/
 

注意不管你使用什么數(shù)據(jù)庫,你都需要安裝相應(yīng)的數(shù)據(jù)庫適配器,每個適配器在網(wǎng)上都是免費的
2,DATABASE_NAME告訴Django數(shù)據(jù)庫名字是什么,如果你使用SQLite,
指出數(shù)據(jù)庫文件的完整的文件系統(tǒng)路徑,如'/home/django/mydata.db'
3,DATABASE_USER告訴Django你連接數(shù)據(jù)庫的用戶名,如果你使用SQLite,這項為空
4,DATABASE_PASSWORD告訴Django你連接數(shù)據(jù)庫的密碼,如果你使用SQLite或者你的密碼為空,則這項為空
5,DATABASE_HOST告訴Django你連接數(shù)據(jù)庫的主機,如果你的數(shù)據(jù)庫和Django安裝在同一臺計算機上,則這項為空
如果你使用SQLite,這項為空
MySQL在這里很特殊,如果這項的值以'/'開頭并且你使用MySQL,MySQL會通過Unix socket連接特殊的socket
例如DATABASE_HOST = '/var/run/mysql/'
如果你使用MySQL但這項的值不是以'/'開頭,那么這項的值就假設(shè)為所連接的主機
6,DATABASE_PORT告訴Django連接數(shù)據(jù)庫的端口,如果你使用SQLite,則這項為空
否則,如果這項為空,底層的數(shù)據(jù)庫適配器會使用給的數(shù)據(jù)庫的默認端口
大部分情況下默認端口即可
一旦你輸入了這項設(shè)置,測試一下你的配置
首先在你第2章創(chuàng)建的mysite項目目錄下運行python manage.py shell
你將會看到進入了Python交互環(huán)境,但是眼睛是會騙人的!
它和普通的python有一個重要的不同,普通的python命令進入的是Python shell,
但是前者告訴Django在啟動shell前使用哪個settings文件
這是做數(shù)據(jù)庫查詢的主要前提,Django需要知道使用哪個settings文件來得到數(shù)據(jù)庫連接信息
在后臺,python manage.py shell設(shè)置了DJANGO_SETTINGS_MODULE環(huán)境變量
后面我們會解釋它的微妙之處,先讓我們測試一下數(shù)據(jù)庫配置:
>>> from django.db import connnection
>>> cursor = connection.cursor()
如果什么事情都沒有發(fā)生,則你的數(shù)據(jù)庫配置對了
否則,檢查錯誤信息作為線索,看看哪里出錯了,下面是一些常見的錯誤:
Java代碼 復(fù)制代碼
  1. 錯誤信息                                                    解決方法    
  2. You haven’t set the DATABASE_ENGINE setting yet.   
  3. 設(shè)置DATABASE_ENGINE而不是為空   
  4.   
  5. Environment variable DJANGO_SETTINGS_MODULE is undefined.   
  6. 運行command python manage.py shell而不是python   
  7.   
  8. Error loading __ module: No module named __.   
  9. 你還沒有安裝數(shù)據(jù)庫相關(guān)的適配器(如psycopg或MySQLdb)   
  10.   
  11. __ isn’t an available database backend.   
  12. 將你的DATABASE_ENGINE設(shè)置為合法的數(shù)據(jù)庫引擎,你是不是敲錯字母了?   
  13.   
  14. database __ does not exist   
  15. 更改DATABASE_NAME指向一個存在的數(shù)據(jù)庫,或者執(zhí)行CREATE DATABASE語句來創(chuàng)建它   
  16.   
  17. role __ does not exist   
  18. 更改DATABASE_USER指向一個存在的user,或者在數(shù)據(jù)庫中創(chuàng)建一個user   
  19.   
  20. could not connect to server   
  21. 確認DATABASE_HOST和DATABASE_PORT設(shè)置正確,以及確認數(shù)據(jù)庫正在運行  


你的第一個app
既然你驗證了數(shù)據(jù)庫連接正確,現(xiàn)在就來創(chuàng)建一個Django app
Django app是一些Django代碼,包括模型和視圖,它們在同一個Python包下面,代表了一個完整的Django程序
在這里值得解釋一下術(shù)語,因為這容易使初學者弄糊涂
我們第2章已經(jīng)創(chuàng)建了一個project,那么project和app的區(qū)別是什么呢?區(qū)別就是配置和代碼:
1,一個project是許多Django app的集合的實例,加上那些app的的配置
技術(shù)上來說,一個project唯一的前提是它提供一個settings文件,里面定義了數(shù)據(jù)庫連接信息,
安裝的app,TEMPLATE_DIRS等等
2,一個app是Django的可移動功能集,通常包括模型和視圖,存在于一個單獨的Python包里面
例如,Django含有幾個app,如commenting系統(tǒng)和自動的admin界面
關(guān)鍵要注意的是它們是可移動并且可以在不同的project重用
沒有嚴格的規(guī)定怎樣安排和計劃你的Django代碼,它是很靈活的
如果你在構(gòu)建一個單獨的網(wǎng)站,你可能只使用一個app
如果你在構(gòu)建一個復(fù)雜的站點,你可能想把它分成幾個app,這樣你就可以在以后分別重用他們
在前面我們的例子中證明我們確實根本不需要創(chuàng)建app,我們只是創(chuàng)建了一個viws.py文件
然后在里面寫視圖方法并設(shè)置我們的URL配置指向這些方法,我們不需要“apps”
但是,有一點需要重視app慣例,如果你使用Django的數(shù)據(jù)庫層(模型),你必須創(chuàng)建Django app
模型必須存在于app,所以為了開始寫模型,我們將創(chuàng)建一個新的app
在前面創(chuàng)建的mysite目錄下面,運行下面的命令來創(chuàng)建一個新的app:
python manage.py startapp books
這個命令不會造成任何輸出,但它在mysite目錄下創(chuàng)建了一個books目錄,讓我們看看它的內(nèi)容:
books/
    __init__.py
    models.py
    views.py
這些文件將包含這個app的模型和視圖
用你最喜歡的文本編輯器看看models.py和views.py,它們都是空的,除了models.py里一個import
這是你的Django app的空白區(qū)

用Python定義模型
我們前面討論到,MTV中的M代表模型
一個Django模型用Python代碼描述了你的數(shù)據(jù)庫中的數(shù)據(jù)
它是你的數(shù)據(jù)結(jié)構(gòu),相當于SQL的CREATE TABLE語句,除了在Python中它比數(shù)據(jù)庫定義包含的內(nèi)容更多
Django在后臺使用模型來執(zhí)行SQL代碼并返回方便的Python數(shù)據(jù)結(jié)構(gòu)來表示你的數(shù)據(jù)庫表的行
Django也使用模型來描述一些高級概念,這些SQL是做不到的
如果你對數(shù)據(jù)庫很熟悉,你可能馬上會想到既在Python中又在SQL中定義數(shù)據(jù)模型豈不是很多余?
Django采用這種工作方式有幾個原因:
1,自省要求過度并且不完美
為了提供方便的數(shù)據(jù)訪問API,Django需要知道數(shù)據(jù)庫結(jié)構(gòu),有兩種方式達到這個目標
一是在Python里顯式的描述數(shù)據(jù),一是運行時內(nèi)省數(shù)據(jù)庫來決定數(shù)據(jù)模型
第二種方式看起來更干凈,因為表的元數(shù)據(jù)僅僅存在于一個地方,但這會導致幾個問題
第一,運行時內(nèi)省數(shù)據(jù)庫顯然要求過度
如果每次Web請求都需要內(nèi)省數(shù)據(jù)庫,即使Web服務(wù)器已經(jīng)初始化,這也會導致的過度的等級不可接受
(有些人認為這個過度的等級可以接受,但Django的開發(fā)者目標是打敗盡可能多的過度框架,
所以這個方案使Django成功的在速度上快于其它的高級框架)第二,一些數(shù)據(jù)庫特別是舊版本的MySQL
并不把足夠的元數(shù)據(jù)存儲起來,所以就導致不能進行準確和完整的自省
2,寫Python代碼是快樂的,保持所有的事情用Python來做可以減少你大腦作切換的時間
如果你保持一個單獨的開發(fā)環(huán)境和心智盡可能久,它將是你非常的高效
寫SQL,然后Python,然后又SQL是很令人心煩的
3,讓數(shù)據(jù)模型存儲在代碼里而不是你的數(shù)據(jù)庫會使你更容易控制你的模型版本
這樣你可以很輕松的跟蹤你的數(shù)據(jù)的更改
4,SQL僅僅允許關(guān)于數(shù)據(jù)結(jié)構(gòu)的某一級別的元數(shù)據(jù)
例如,大部分數(shù)據(jù)庫系統(tǒng)并不提供專門的數(shù)據(jù)類型來支持e-mail地址或者url
Django模型則可以,高級數(shù)據(jù)類型的優(yōu)點是更高的生產(chǎn)率和更易重用的代碼
5,SQL在不同的數(shù)據(jù)庫平臺不一致,例如,如果你正在發(fā)布你一個Web程序
發(fā)布一個Python模塊來描述數(shù)據(jù)結(jié)構(gòu)會比分開為MySQL,PostgreSQL和SQLite寫CREATE TABLE語句更高效
盡管如此,這個方法的一個缺點是Python代碼所做的事情可能超出實際上數(shù)據(jù)庫里的數(shù)據(jù)的范圍
如果你更改了Django模型,你需要在你的數(shù)據(jù)庫做同樣的改動做保持數(shù)據(jù)庫和模型一致
本章后面我們將詳細解釋解決此問題的策略
最后,我們必須指出的是Django包含了一個輔助工具來通過現(xiàn)存的數(shù)據(jù)庫生成模型
這對于迅速接管和運行遺留數(shù)據(jù)很有幫助

你的第一個模型
這一章我們將關(guān)注book/author/publisher數(shù)據(jù)結(jié)構(gòu),它們是眾所周知的
我們將支持一下概念,域和關(guān)系:
1,一個author有一個salutation(如Mr.或Mrs.),一個first name,一個last name,一個e-mail地址和一個頭像photo
2,一個publisher有一個name,一個street地址,一個city,一個state/province,一個country和一個Web site
3,一個book有一個title和一個publication date,一個或多個authors(many-to-many),一個單獨的publisher(one-to-many)
在Django中第一步是使用Python代碼描述上面的數(shù)據(jù)庫結(jié)構(gòu),在startapp命令創(chuàng)建的models.py中輸入下面的內(nèi)容:
Java代碼 復(fù)制代碼
  1. from django.db import models   
  2.   
  3. class Publisher(models.Model):   
  4.     name = models.CharField(maxlength=30)   
  5.     address = models.CharField(maxlength=50)   
  6.     city = models.CharField(maxlength=60)   
  7.     state_province = models.CharField(maxlength=30)   
  8.     country = models.CharField(maxlength=50)   
  9.     website = models.URLField()   
  10.   
  11. class Author(models.Model):   
  12.     salutation = models.CharField(maxlength=10)   
  13.     first_name = models.CharField(maxlength=30)   
  14.     last_name = models.CharField(maxlength=40)   
  15.     email = models.EmailField()   
  16.     headshot = models.ImageField(upload_to='/tmp')   
  17.   
  18. class Book(models.Model):   
  19.     title = models.CharField(maxlength=100)   
  20.     authors = models.ManyToManyField(Author)   
  21.     publisher = models.ForeignKey(Publisher)   
  22.     publication_date = models.DateField()  

這章我們會談到模型語法和選項,讓我們先來快速的看看這些代碼來得到基本的印象
要注意的第一點是每個模型都是django.db.models.Model的子類
它們的父類Model包含了讓這些對象具有與數(shù)據(jù)庫交互能力的機制
這樣一來我們的模型只負責定義自己的域就行了,語法相當簡潔緊湊
不管相信與否,這就是通過Django進行數(shù)據(jù)訪問的所有代碼
一個模型通常域一個數(shù)據(jù)庫表對應(yīng),而每個屬性和數(shù)據(jù)庫表的一列對應(yīng)
屬性名對應(yīng)列名,屬性的類型(如CharField)對應(yīng)數(shù)據(jù)庫列類型
例如Publisher模型對應(yīng)了下面的表(假設(shè)使用PostgreSQL的CREATE TABLE語法):
Java代碼 復(fù)制代碼
  1. CREATE TABLE "books_publisher" (   
  2.     "id" serial NOT NULL PRIMARY KEY,   
  3.     "name" varchar(30) NOT NULL,   
  4.     "address" varchar(50) NOT NULL,   
  5.     "city" varchar(60) NOT NULL,   
  6.     "state_province" varchar(30) NOT NULL,   
  7.     "country" varchar(50) NOT NULL,   
  8.     "website" varchar(200) NOT NULL   
  9. );  

事實上Django自己可以生成CREAT TABLE語句,我們一會再看
一個類對應(yīng)一個數(shù)據(jù)庫表的特例是多對多關(guān)系,我們的例子中Book有一個ManyToManyField叫作authors
這表明book擁有一個或多個authors,但是Book表并沒有authors列
Django創(chuàng)建了一個附加的多對多連接表來處理books到authors的映射
最后注意的是我們沒有在任何一個模型中顯示的定義主鍵
除非你自己定義一個主鍵,Django會自動為每個模型生成一個integer主鍵域id
每個Django模型都必須有一個單列的主鍵

安裝模型
寫完代碼,下面讓我們來創(chuàng)建數(shù)據(jù)庫表
第一步是在Django中激活這些模型,需要把books這個app添加到settings文件的apps列表
編輯settings.py,查找INSTALLED_APPS設(shè)置
INSTALLED_APPS告訴Django哪些apps是活動的,默認時如下所示:
Java代碼 復(fù)制代碼
  1. INSTALLED_APPS = (   
  2.     'django.contrib.auth',   
  3.     'django.contrib.contenttypes',   
  4.     'django.contrib.sessions',   
  5.     'django.contrib.sites',   
  6. )  

先用(#)把這些strings注釋掉,后面我們再激活和討論它們
然后添加'mysite.books'到INSTALLED_APPS列表,最后如下所示:
Java代碼 復(fù)制代碼
  1. INSTALLED_APPS = (   
  2.     #'django.contrib.auth',   
  3.     #'django.contrib.contenttypes',   
  4.     #'django.contrib.sessions',   
  5.     #'django.contrib.sites',   
  6.     'mysite.books',   
  7. )  

別忘了最后的逗號
順便說一下,本書作者習慣與在元組的元素后面都加上逗號,無論元組是否只有一個元素
這可以避免忘記加逗號,加了也不會罰款
'mysite.books'指我們正在工作的books app
INSTALLED_APPS中的每個app都用完整的Python PATH來表示,即包的PATH,用小數(shù)點分隔來指向app包
Django app已經(jīng)在settings文件激活,我們可以在數(shù)據(jù)庫中創(chuàng)建表了
首先通過如下的命令驗證一下模型:python manage.py validate
validate命令檢查我們的模型語法和邏輯正確與否
如果一切正常,我們會看到0 errors found的信息
否則,確認你的模型代碼輸入正確,error輸出會給你有用的信息來幫你找到錯誤的代碼
任何時候你認為你的模型代碼有問題都可以運行python manage.py validate來捕捉模型錯誤
如果你的模型是合法的,運行下面的命令為books app的模型生成CREATE TABLE語句
(如果你使用Unix會有五顏六色的語法高亮):python manage.py sqlall books
這個命令中,books是app的名字,運行完命令,你會看到下面的信息:
Java代碼 復(fù)制代碼
  1. BEGIN;   
  2. CREATE TABLE "books_publisher" (   
  3.     "id" serial NOT NULL PRIMARY KEY,   
  4.     "name" varchar(30) NOT NULL,   
  5.     "address" varchar(50) NOT NULL,   
  6.     "city" varchar(60) NOT NULL,   
  7.     "state_province" varchar(30) NOT NULL,   
  8.     "country" varchar(50) NOT NULL,   
  9.     "website" varchar(200) NOT NULL   
  10. );   
  11. CREATE TABLE "books_book" (   
  12.     "id" serial NOT NULL PRIMARY KEY,   
  13.     "title" varchar(100) NOT NULL,   
  14.     "publisher_id" integer NOT NULL REFERENCES "books_publisher" ("id"),   
  15.     "publication_date" date NOT NULL   
  16. );   
  17. CREATE TABLE "books_author" (   
  18.     "id" serial NOT NULL PRIMARY KEY,   
  19.     "salutation" varchar(10) NOT NULL,   
  20.     "first_name" varchar(30) NOT NULL,   
  21.     "last_name" varchar(40) NOT NULL,   
  22.     "email" varchar(75) NOT NULL,   
  23.     "headshot" varchar(100) NOT NULL   
  24. );   
  25. CREATE TABLE "books_book_authors" (   
  26.     "id" serial NOT NULL PRIMARY KEY,   
  27.     "book_id" integer NOT NULL REFERENCES "books_book" ("id"),   
  28.     "author_id" integer NOT NULL REFERENCES "books_author" ("id"),   
  29.     UNIQUE ("book_id""author_id")   
  30. );   
  31. CREATE INDEX books_book_publisher_id ON "books_book" ("publisher_id");   
  32. COMMIT;  

注意以下幾點:
1,表明自動由app名(books)和小寫的模型名-publisher,book和author組成
你可以覆蓋這個行為,我們本章后面會看到
2,前面提到,Django自動給每個表添加主鍵id域,你也可以覆蓋這點
3,習慣約束上Django會在外鍵域的名字后面添加“_id”,你已經(jīng)猜到了,你也可以覆蓋這點
4,外鍵關(guān)系由顯式的REFERENCES語句來完成
5,這些CREATE TABLE語句是針對你使用的數(shù)據(jù)庫生成的,所以數(shù)據(jù)庫專有的域類型如
aotu_increment(MySQL),serial(PostgreSQL),或者integer primary key(SQLite)會自動為你處理
類似的如表名的引號是使用單引號還是雙引號也一樣,這個例子是使用的PostgreSQL語法
sqlall命令事實上并沒有接觸數(shù)據(jù)庫或建表,它僅僅將輸出打印到屏幕上
所以如果你問它,你可以看到DJango將執(zhí)行什么
如果你愿意,你可以復(fù)制粘貼這些SQL到你數(shù)據(jù)庫客戶端或者使用Unix管道來直接傳遞它
盡管如此,Django提供一個簡單的方式來把這些SQL提交數(shù)據(jù)庫
像下面這樣運行syncdb命令:python manage.py syncdb
你會看到如下信息:
Creating table books_publisher
Creating table books_book
Creating table books_author
Installing index for books.Book model
syncdb簡單的把你的模型同步到數(shù)據(jù)庫
它檢查數(shù)據(jù)庫和你的INSTALLED_APPS中的所有app的所以模型,看看是否有些表已經(jīng)存在,如果表不存在就創(chuàng)建表
注意syncdb不會同步改動或刪除了的模型,如果你改動或刪除了一個模型,syncdb不會更新數(shù)據(jù)庫(待會兒討論這個)
如果你再運行一次python manage.py syncdb,不會發(fā)生任何事情
因為你沒有添加模型到books app或添加到INSTALLED_APPS中的任何app
因此運行python manage.py syncdb是一直安全的,它不會把事情弄糟
如果你感興趣,進入你的數(shù)據(jù)庫服務(wù)器的命令行客戶端看看Django創(chuàng)建的數(shù)據(jù)庫表
你可以手動運行命令行客戶端如PostgreSQL的psql,或者運行python manage.py dbshell
基于你的DATABASE_SERVER設(shè)置,后者將計算出運行哪個命令行客戶端,也更方便

數(shù)據(jù)訪問基礎(chǔ)
一旦你創(chuàng)建了一個模型,Django自動提供高級Python API給這些模型工作
運行python manage.py shell然后輸入下面的代碼試試:
Java代碼 復(fù)制代碼
  1. >>> from books.models import Publisher   
  2. >>> p = Publisher(name='Apress', address='2560 Ninth St.',   
  3. ...     city='Berkeley', state_province='CA', country='U.S.A.',   
  4. ...     website='http://www.apress.com/')   
  5. >>> p.save()   
  6. >>> p = Publisher(name="O'Reilly", address='10 Fawcett St.',   
  7. ...     city='Cambridge', state_province='MA', country='U.S.A.',   
  8. ...     website='http://www.oreilly.com/')   
  9. >>> p.save()   
  10. >>> publisher_list = Publisher.objects.all()   
  11. >>> publisher_list   
  12. [<Publisher: Publisher object>, <Publisher: Publisher object>]  

雖然只有幾行代碼,確達到了很多目的,精彩的部分是:
1,創(chuàng)建一個對象只需import合適的模型類并通過給每個域傳遞值來初始化它
2,調(diào)用save()方法來將一個對象保存到數(shù)據(jù)庫,后臺Django在這里執(zhí)行了一條INSERT SQL語句
3,使用Publisher.objects屬性從數(shù)據(jù)庫得到對象,使用Publisher.objects.all()得到Publisher所有的對象列表
后臺Django在這里執(zhí)行了一條SELECT SQL語句
實際上你可以通過Django數(shù)據(jù)庫API做很多事情,但是我們先來看一個小麻煩

添加模型的string顯示
上面的例子中,當我們打印publishers列表時我們得到的都是一些無用的信息,我們很難將Publisher對象區(qū)別開:
[<Publisher: Publisher object>, <Publisher: Publisher object>]
我們可以通過給Publisher對象添加一個__str__()方法來輕松解決這個問題
__str__()方法告訴Python怎樣顯示對象的string顯示,你可以動手來看看給三個模型添加__str__():
Java代碼 復(fù)制代碼
  1. class Publisher(models.Model):   
  2.     name = models.CharField(maxlength=30)   
  3.     address = models.CharField(maxlength=50)   
  4.     city = models.CharField(maxlength=60)   
  5.     state_province = models.CharField(maxlength=30)   
  6.     country = models.CharField(maxlength=50)   
  7.     website = models.URLField()   
  8.   
  9.     def __str__(self):   
  10.         return self.name   
  11.   
  12. class Author(models.Model):   
  13.     salutation = models.CharField(maxlength=10)   
  14.     first_name = models.CharField(maxlength=30)   
  15.     last_name = models.CharField(maxlength=40)   
  16.     email = models.EmailField()   
  17.     headshot = models.ImageField(upload_to='/tmp')   
  18.   
  19.     def __str__(self):   
  20.         return '%s %s' % (self.first_name, self.last_name)   
  21.   
  22. class Book(models.Model):   
  23.     title = models.CharField(maxlength=100)   
  24.     authors = models.ManyToManyField(Author)   
  25.     publisher = models.ForeignKey(Publisher)   
  26.     publication_date = models.DateField()   
  27.   
  28.     def __str__(self):   
  29.         return self.title  

你可以看到,__str__()方法為了返回一個string顯示可以做任何它需要做的事情
這里Publisher和Book的__str__()方法簡單的返回了對象的name和title
但是Author的__str__()更復(fù)雜一點,返回了first_name和last_name的組合
__str__()唯一的條件是返回一個string,如果不返回string的話如返回一個integer
Python會觸發(fā)一個TypeError異常,并帶有“__str__ returned non-string”信息
為了讓改動生效,退出Python然后使用python manage.py shell命令重新進入
(這是讓代碼改動生效的最簡單的方式)
現(xiàn)在,Publisher列表對象更容易理解:
Java代碼 復(fù)制代碼
  1. >>> from books.models import Publisher   
  2. >>> publisher_list = Publisher.objects.all()   
  3. >>> publisher_list   
  4. [<Publisher: Apress>, <Publisher: O'Reilly>]  

確認你定義的任何模型都有一個__str__()方法,不僅是使在你自己使用交互環(huán)境時更方便
也因為當Django在幾個地方需要顯示對象時會使用__str__()的輸出
最后,注意__str__()是給模型添加行為的好習慣
一個Django模型描述的不僅僅是一個對象數(shù)據(jù)庫表結(jié)構(gòu),它也描述了對象知道怎樣去做的功能
__str__()就是這樣的功能的一個例子,一個模型知道怎樣顯示它自己


安徽新華電腦學校專業(yè)職業(yè)規(guī)劃師為你提供更多幫助【在線咨詢
国产极品白嫩美女在线观看看| 高清一级毛片一本到免费观看| 国产91精品系列在线观看| 日韩一级黄色| 99热精品在线| 亚欧成人乱码一区二区| 九九九网站| 久久精品大片| 九九精品在线播放| 成人高清视频免费观看| 精品视频免费观看| 国产欧美精品午夜在线播放| 麻豆网站在线看| 日本伦理网站| 青青久久精品国产免费看| 欧美一区二区三区性| 二级片在线观看| 999精品视频在线| 九九久久国产精品| 亚洲精品久久久中文字| 日本免费区| a级黄色毛片免费播放视频| 国产一区二区精品| 免费一级片在线| 日韩中文字幕在线亚洲一区| 久久99爰这里有精品国产| 午夜激情视频在线观看| 99色吧| 99久久精品国产麻豆| 亚久久伊人精品青青草原2020| 欧美大片一区| 一本伊大人香蕉高清在线观看| 欧美另类videosbestsex久久| 亚飞与亚基在线观看| 日韩字幕在线| 四虎久久影院| 日韩中文字幕一区| 久久国产一区二区| 国产精品免费久久| 久久久久久久免费视频| 久久国产精品自线拍免费| 四虎久久影院| 午夜激情视频在线播放| 成人免费网站久久久| 一级毛片看真人在线视频| 久久福利影视| 久久成人性色生活片| 国产视频久久久久| 日韩中文字幕一区| 国产成人啪精品| 国产高清在线精品一区a| 日韩中文字幕一区二区不卡| 欧美日本免费| 国产伦理精品| 精品久久久久久中文字幕2017| 日韩av东京社区男人的天堂| 精品美女| 欧美一级视| 999精品在线| 韩国三级视频在线观看| 日韩中文字幕在线亚洲一区| 国产伦精品一区二区三区在线观看 | 欧美夜夜骑 青草视频在线观看完整版 久久精品99无色码中文字幕 欧美日韩一区二区在线观看视频 欧美中文字幕在线视频 www.99精品 香蕉视频久久 | 国产伦精品一区三区视频| 欧美激情一区二区三区视频 | 欧美激情伊人| 国产精品12| 一级女性全黄生活片免费| 毛片的网站| 国产不卡福利| 免费的黄色小视频| 亚洲女人国产香蕉久久精品| 欧美激情一区二区三区在线播放| 二级片在线观看| 一本伊大人香蕉高清在线观看| 国产欧美精品午夜在线播放| 精品国产亚洲人成在线| 久久99这里只有精品国产| 午夜家庭影院| 天天做日日爱夜夜爽| 好男人天堂网 久久精品国产这里是免费 国产精品成人一区二区 男人天堂网2021 男人的天堂在线观看 丁香六月综合激情 | 一本高清在线| 你懂的福利视频| 欧美激情伊人| 可以免费看污视频的网站| 日本在线www| 精品国产香蕉在线播出| 日本免费乱人伦在线观看 | 成人高清视频免费观看| 日本伦理片网站| 久久成人性色生活片| 国产成人精品在线| 美国一区二区三区| 国产麻豆精品视频| 午夜欧美福利| 国产视频一区二区在线播放| 国产91精品系列在线观看| 九九干| 欧美一级视频免费观看| 成人免费福利片在线观看| 黄色短视屏| 国产高清在线精品一区a| 国产一区精品| 亚欧成人乱码一区二区| 欧美a级片视频| 国产综合成人观看在线| 国产亚洲免费观看| 日韩欧美一及在线播放| 成人a大片高清在线观看| 欧美a免费| 国产精品12| 天天做人人爱夜夜爽2020毛片| 97视频免费在线观看| 欧美一级视| 国产视频一区二区在线观看| 精品美女| 国产一区二区精品久久91| 99色吧| 色综合久久久久综合体桃花网| 四虎影视久久久| 国产美女在线观看| 亚洲天堂一区二区三区四区| 精品久久久久久免费影院| 国产伦理精品| 精品视频在线观看免费 | 欧美国产日韩精品| 欧美一级视| 一级片免费在线观看视频| 亚洲不卡一区二区三区在线| 成人av在线播放| 国产精品免费久久| 欧美大片aaaa一级毛片| 天堂网中文在线| 亚洲第一视频在线播放| 青青久热| 国产不卡福利| 国产高清视频免费观看| 四虎影视库国产精品一区| 欧美夜夜骑 青草视频在线观看完整版 久久精品99无色码中文字幕 欧美日韩一区二区在线观看视频 欧美中文字幕在线视频 www.99精品 香蕉视频久久 | 久久久久久久男人的天堂| 国产国语对白一级毛片| 亚洲天堂一区二区三区四区| 欧美1卡一卡二卡三新区| 韩国三级香港三级日本三级| 成人影视在线播放| 日韩免费片| 国产伦精品一区二区三区在线观看 | 国产伦精品一区二区三区无广告 | 一级女性全黄久久生活片| 精品国产一级毛片| 韩国毛片免费| 成人影院一区二区三区| 一级片免费在线观看视频| 国产麻豆精品视频| 欧美夜夜骑 青草视频在线观看完整版 久久精品99无色码中文字幕 欧美日韩一区二区在线观看视频 欧美中文字幕在线视频 www.99精品 香蕉视频久久 | 国产成人精品综合在线| 99久久精品费精品国产一区二区| 欧美激情一区二区三区在线| 国产麻豆精品免费密入口| 日韩免费在线视频| 久草免费在线视频| 欧美激情一区二区三区视频 | 99色视频在线| 国产亚洲免费观看| 国产成人精品综合| 在线观看导航| 久久精品欧美一区二区| 亚洲精品中文一区不卡| 久久国产影院| 国产伦精品一区二区三区无广告| 一级女性大黄生活片免费| 99久久精品国产国产毛片 | 国产麻豆精品| 日本久久久久久久 97久久精品一区二区三区 狠狠色噜噜狠狠狠狠97 日日干综合 五月天婷婷在线观看高清 九色福利视频 | 欧美激情一区二区三区视频| 欧美激情一区二区三区视频 | 亚洲 激情| 日本久久久久久久 97久久精品一区二区三区 狠狠色噜噜狠狠狠狠97 日日干综合 五月天婷婷在线观看高清 九色福利视频 | 一级女性全黄生活片免费 | 成人影院久久久久久影院| 亚洲精品久久玖玖玖玖| 久久国产一区二区| 91麻豆精品国产自产在线观看一区 | 欧美爱爱动态| 人人干人人插| 国产原创中文字幕| 国产激情视频在线观看| 青青久久精品国产免费看| 欧美激情伊人| 亚洲 激情| 国产视频一区二区三区四区| 欧美国产日韩在线| 日韩在线观看免费| 国产韩国精品一区二区三区| 成人a级高清视频在线观看| a级精品九九九大片免费看| 国产一区免费观看| 日韩在线观看视频黄| 成人av在线播放| 国产亚洲免费观看|