国产精品一区二区精品_久久小视频_天堂va在线观看_99久久夜色精品国产亚洲96_日本手机在线视频_av成人免费

當(dāng)前位置:首頁 > 網(wǎng)站舊欄目 > 學(xué)習(xí)園地 > 設(shè)計(jì)軟件教程 > 翻譯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配置
如我們所說,試圖負(fù)責(zé)邏輯和返回應(yīng)答,例子中我們計(jì)算了當(dāng)前的日期和時(shí)間
現(xiàn)在的Web程序中常常和數(shù)據(jù)庫打交道
一個(gè)數(shù)據(jù)庫驅(qū)動的網(wǎng)站在后臺連接數(shù)據(jù)庫服務(wù)器,得到并顯示很好的格式化的Web頁面
同樣,網(wǎng)站也可以提供給訪問者也具有操作數(shù)據(jù)庫的功能
許多復(fù)雜的網(wǎng)站以上兩種功能的結(jié)合,如Amazon.com就是一個(gè)數(shù)據(jù)庫驅(qū)動的站點(diǎn)
每一個(gè)產(chǎn)品頁面都是Amazon數(shù)據(jù)庫格式后的HTML,你訪問頁面也就是間接訪問數(shù)據(jù)庫
Django很適合數(shù)據(jù)庫驅(qū)動的網(wǎng)站,通過Python它提供強(qiáng)大的數(shù)據(jù)庫訪問能力
這章將講述Django的數(shù)據(jù)庫層

在視圖里進(jìn)行數(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})  

這個(gè)方法可以工作,但是馬上一些問題出來了:
1,我們把數(shù)據(jù)庫連接的參數(shù)硬編碼到代碼里面了,理想狀況下它們應(yīng)該存儲在Django配置里面
2,我們必須寫一些樣板文件代碼,如建立連接,創(chuàng)建cursor,執(zhí)行語句和關(guān)閉連接等
細(xì)想狀況下,我們應(yīng)該只需指出我們需要什么結(jié)果
3,它把我們和MySQL綁在一起,如果我們想切換到PostgreSQL
我們必須使用不同的數(shù)據(jù)庫適配器psycopg,改變數(shù)據(jù)庫連接參數(shù)以及考慮重寫SQL語句
理想狀況下數(shù)據(jù)庫服務(wù)器應(yīng)該是抽象的,替換數(shù)據(jù)庫應(yīng)該只在一個(gè)地方設(shè)置
你可能會想,Django的數(shù)據(jù)庫層的目標(biāo)應(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ā)模式
在我們專研更多的代碼之前,讓我們先花點(diǎn)時(shí)間考慮一些Django Web程序的整體設(shè)計(jì)
前面的章節(jié)我們提到,Django設(shè)計(jì)來鼓勵松耦合和分離程序模塊
如果你遵循這個(gè)哲學(xué),改變一部分代碼而不影響其它模塊是很容易做到的
例如在視圖方法里,我們討論了使用模板來分離業(yè)務(wù)邏輯和呈現(xiàn)邏輯的重要性
在數(shù)據(jù)庫層的數(shù)據(jù)訪問邏輯我們將遵循同樣的哲學(xué)
數(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)中通過用戶輸入決定使用哪個(gè)視圖及訪問必要的模型的部分
采用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)用來處理
因?yàn)?ldquo;C”是Django框架本身處理而導(dǎo)致Django大部分精彩的東西在于模型,模板和視圖
所以Django被稱位MTV框架:
1,M,代表模型,是數(shù)據(jù)訪問層,它包含了關(guān)于數(shù)據(jù)的一切東西,怎樣得到數(shù)據(jù),怎樣驗(yàn)證數(shù)據(jù),
它具有什么行為以及數(shù)據(jù)之間的關(guān)系
2,T,代表模板,是展現(xiàn)層,它包含了呈現(xiàn)相關(guān)的決策,如內(nèi)容怎樣在Web頁面中顯示以及其它類型的文檔
3,V,代表視圖,是業(yè)務(wù)邏輯層,它包含了訪問模型的邏輯和選擇合適的模板
你可以認(rèn)為視圖是模型和模板的橋梁
如果你對MVC框架熟悉,如Ruby on Rails,你可以把Django的視圖想象成“controllers”,
把Django的模板想象成“views”,這是對MVC的不同解釋造成的不幸的混亂
在Django關(guān)于MVC的解釋中,“view”描述呈現(xiàn)給用戶的數(shù)據(jù)
沒有必要弄清數(shù)據(jù)怎樣顯示,而是描述哪個(gè)數(shù)據(jù)應(yīng)該被呈現(xiàn)
對比而言,Ruby on Rails以及類似的框架建議controller的工作包括決定哪個(gè)數(shù)據(jù)顯示給用戶,
視圖嚴(yán)格的決定數(shù)據(jù)怎樣顯示,而不是決定哪個(gè)數(shù)據(jù)來顯示
每一個(gè)解釋都不比另一個(gè)正確,最重要的事情是理解底層的概念

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

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


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

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

你的第一個(gè)模型
這一章我們將關(guān)注book/author/publisher數(shù)據(jù)結(jié)構(gòu),它們是眾所周知的
我們將支持一下概念,域和關(guān)系:
1,一個(gè)author有一個(gè)salutation(如Mr.或Mrs.),一個(gè)first name,一個(gè)last name,一個(gè)e-mail地址和一個(gè)頭像photo
2,一個(gè)publisher有一個(gè)name,一個(gè)street地址,一個(gè)city,一個(gè)state/province,一個(gè)country和一個(gè)Web site
3,一個(gè)book有一個(gè)title和一個(gè)publication date,一個(gè)或多個(gè)authors(many-to-many),一個(gè)單獨(dú)的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()  

這章我們會談到模型語法和選項(xiàng),讓我們先來快速的看看這些代碼來得到基本的印象
要注意的第一點(diǎn)是每個(gè)模型都是django.db.models.Model的子類
它們的父類Model包含了讓這些對象具有與數(shù)據(jù)庫交互能力的機(jī)制
這樣一來我們的模型只負(fù)責(zé)定義自己的域就行了,語法相當(dāng)簡潔緊湊
不管相信與否,這就是通過Django進(jìn)行數(shù)據(jù)訪問的所有代碼
一個(gè)模型通常域一個(gè)數(shù)據(jù)庫表對應(yīng),而每個(gè)屬性和數(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. );  

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

安裝模型
寫完代碼,下面讓我們來創(chuàng)建數(shù)據(jù)庫表
第一步是在Django中激活這些模型,需要把books這個(gè)app添加到settings文件的apps列表
編輯settings.py,查找INSTALLED_APPS設(shè)置
INSTALLED_APPS告訴Django哪些apps是活動的,默認(rèn)時(shí)如下所示:
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. )  

別忘了最后的逗號
順便說一下,本書作者習(xí)慣與在元組的元素后面都加上逗號,無論元組是否只有一個(gè)元素
這可以避免忘記加逗號,加了也不會罰款
'mysite.books'指我們正在工作的books app
INSTALLED_APPS中的每個(gè)app都用完整的Python PATH來表示,即包的PATH,用小數(shù)點(diǎn)分隔來指向app包
Django app已經(jīng)在settings文件激活,我們可以在數(shù)據(jù)庫中創(chuàng)建表了
首先通過如下的命令驗(yàn)證一下模型:python manage.py validate
validate命令檢查我們的模型語法和邏輯正確與否
如果一切正常,我們會看到0 errors found的信息
否則,確認(rèn)你的模型代碼輸入正確,error輸出會給你有用的信息來幫你找到錯(cuò)誤的代碼
任何時(shí)候你認(rèn)為你的模型代碼有問題都可以運(yùn)行python manage.py validate來捕捉模型錯(cuò)誤
如果你的模型是合法的,運(yùn)行下面的命令為books app的模型生成CREATE TABLE語句
(如果你使用Unix會有五顏六色的語法高亮):python manage.py sqlall books
這個(gè)命令中,books是app的名字,運(yùn)行完命令,你會看到下面的信息:
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;  

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

數(shù)據(jù)訪問基礎(chǔ)
一旦你創(chuàng)建了一個(gè)模型,Django自動提供高級Python API給這些模型工作
運(yùn)行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>]  

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

添加模型的string顯示
上面的例子中,當(dāng)我們打印publishers列表時(shí)我們得到的都是一些無用的信息,我們很難將Publisher對象區(qū)別開:
[<Publisher: Publisher object>, <Publisher: Publisher object>]
我們可以通過給Publisher對象添加一個(gè)__str__()方法來輕松解決這個(gè)問題
__str__()方法告訴Python怎樣顯示對象的string顯示,你可以動手來看看給三個(gè)模型添加__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__()方法為了返回一個(gè)string顯示可以做任何它需要做的事情
這里Publisher和Book的__str__()方法簡單的返回了對象的name和title
但是Author的__str__()更復(fù)雜一點(diǎn),返回了first_name和last_name的組合
__str__()唯一的條件是返回一個(gè)string,如果不返回string的話如返回一個(gè)integer
Python會觸發(fā)一個(gè)TypeError異常,并帶有“__str__ returned non-string”信息
為了讓改動生效,退出Python然后使用python manage.py shell命令重新進(jìn)入
(這是讓代碼改動生效的最簡單的方式)
現(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>]  

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


安徽新華電腦學(xué)校專業(yè)職業(yè)規(guī)劃師為你提供更多幫助【在線咨詢
主站蜘蛛池模板: 国产伦精品一区二区 | 久草在线网址 | 欧美一级大片在线观看 | 午夜激情福利在线 | 国产日韩亚洲欧美 | 国产黄色免费在线观看 | 高清一区二区三区 | 欧美特级 | 黄色av片三级三级三级免费看 | 国产精品av一区二区三区 | 天堂电影在线观看 | 日韩精品电影一区二区 | 黄a大片| 亚洲伊人精品 | 91国拍精品国产粉嫩亚洲一区 | 特级毛片免费 | 日本美女高潮视频 | 色频 | 国产日韩在线 | 亚洲精品播放 | 午夜小视频在线观看 | 特一级黄色 | 国产毛片777777| 性爱视频免费 | 插女人逼视频 | 成人欧美 | 91精品一区二区中文字幕 | 国产午夜在线 | 天天色天天射综合网 | 日韩免费播放 | 一区二区三区在线视频观看 | 久草在线手机 | 精品久久久久久久久 | 偷拍自拍第二页 | 97婷婷狠狠成为人免费视频 | 91插插插插插插插插 | 7788色淫网站免费观看 | 97插插| 日韩黄色特级片 | 国产三级精品三级 | 日韩不卡av在线 |