Django多數(shù)據(jù)庫的實(shí)現(xiàn)過程詳解
有些項(xiàng)目可能涉及到使用多個(gè)數(shù)據(jù)庫的情況,方法很簡單。
1.在settings中設(shè)定DATABASE
比如要使用兩個(gè)數(shù)據(jù)庫:
DATABASES = {
'default': {
'NAME': 'app_data',
'ENGINE': 'django.db.backends.postgresql',
'USER': 'postgres_user',
'PASSWORD': 's3krit'
},
'users': {
'NAME': 'user_data',
'ENGINE': 'django.db.backends.mysql',
'USER': 'mysql_user',
'PASSWORD': 'priv4te'
}
}
這樣就確定了2個(gè)數(shù)據(jù)庫,別名一個(gè)為default,一個(gè)為user。數(shù)據(jù)庫的別名可以任意確定。
default的別名比較特殊,一個(gè)Model在路由中沒有特別選擇時(shí),默認(rèn)使用default數(shù)據(jù)庫。
當(dāng)然,default也可以設(shè)置為空:
DATABASES = {
'default': {},
'users': {
'NAME': 'user_data',
'ENGINE': 'django.db.backends.mysql',
'USER': 'mysql_user',
'PASSWORD': 'superS3cret'
},
'customers': {
'NAME': 'customer_data',
'ENGINE': 'django.db.backends.mysql',
'USER': 'mysql_cust',
'PASSWORD': 'veryPriv@ate'
}
}
這樣,因?yàn)闆]有了默認(rèn)的數(shù)據(jù)庫,就需要為所有的Model,包括使用的第三方庫中的Model做好數(shù)據(jù)庫路由選擇。
2.為需要做出數(shù)據(jù)庫選擇的Model規(guī)定app_label
class MyUser(models.Model):
...
class Meta:
app_label = 'users'
3.寫Database Routers
Database Router用來確定一個(gè)Model使用哪一個(gè)數(shù)據(jù)庫,主要定義以下四個(gè)方法:
db_for_read(model, **hints)
規(guī)定model使用哪一個(gè)數(shù)據(jù)庫讀取。
db_for_write(model, **hints)
規(guī)定model使用哪一個(gè)數(shù)據(jù)庫寫入。
allow_relation(obj1, obj2, **hints)
確定obj1和obj2之間是否可以產(chǎn)生關(guān)聯(lián), 主要用于foreign key和 many to many操作。
allow_migrate(db, app_label, model_name=None, **hints)
確定migrate操作是否可以在別名為db的數(shù)據(jù)庫上運(yùn)行。
一個(gè)完整的例子:
數(shù)據(jù)庫設(shè)定:
DATABASES = {
'default': {},
'auth_db': {
'NAME': 'auth_db',
'ENGINE': 'django.db.backends.mysql',
'USER': 'mysql_user',
'PASSWORD': 'swordfish',
},
'primary': {
'NAME': 'primary',
'ENGINE': 'django.db.backends.mysql',
'USER': 'mysql_user',
'PASSWORD': 'spam',
},
'replica1': {
'NAME': 'replica1',
'ENGINE': 'django.db.backends.mysql',
'USER': 'mysql_user',
'PASSWORD': 'eggs',
},
'replica2': {
'NAME': 'replica2',
'ENGINE': 'django.db.backends.mysql',
'USER': 'mysql_user',
'PASSWORD': 'bacon',
},
}
如果想要達(dá)到如下效果:
app_label為auth的Model讀寫都在auth_db中完成,其余的Model寫入在primary中完成,讀取隨機(jī)在replica1和replica2中完成。
auth:
class AuthRouter(object):
"""
A router to control all database operations on models in the
auth application.
"""
def db_for_read(self, model, **hints):
"""
Attempts to read auth models go to auth_db.
"""
if model._meta.app_label == 'auth':
return 'auth_db'
return None
def db_for_write(self, model, **hints):
"""
Attempts to write auth models go to auth_db.
"""
if model._meta.app_label == 'auth':
return 'auth_db'
return None
def allow_relation(self, obj1, obj2, **hints):
"""
Allow relations if a model in the auth app is involved.
"""
if obj1._meta.app_label == 'auth' or \
obj2._meta.app_label == 'auth':
return True
return None
def allow_migrate(self, db, app_label, model_name=None, **hints):
"""
Make sure the auth app only appears in the 'auth_db'
database.
"""
if app_label == 'auth':
return db == 'auth_db'
return None
這樣app_label為auth的Model讀寫都在auth_db中完成,允許有關(guān)聯(lián),migrate只在auth_db數(shù)據(jù)庫中可以運(yùn)行。
其余的:
import random
class PrimaryReplicaRouter(object):
def db_for_read(self, model, **hints):
"""
Reads go to a randomly-chosen replica.
"""
return random.choice(['replica1', 'replica2'])
def db_for_write(self, model, **hints):
"""
Writes always go to primary.
"""
return 'primary'
def allow_relation(self, obj1, obj2, **hints):
"""
Relations between objects are allowed if both objects are
in the primary/replica pool.
"""
db_list = ('primary', 'replica1', 'replica2')
if obj1._state.db in db_list and obj2._state.db in db_list:
return True
return None
def allow_migrate(self, db, app_label, model_name=None, **hints):
"""
All non-auth models end up in this pool.
"""
return True
這樣讀取在隨機(jī)在replica1和replica2中完成,寫入使用primary。
最后在settings中設(shè)定:
DATABASE_ROUTERS = ['path.to.AuthRouter', 'path.to.PrimaryReplicaRouter']
就可以了。
進(jìn)行migrate操作時(shí):
$ ./manage.py migrate $ ./manage.py migrate --database=users
migrate操作默認(rèn)對default數(shù)據(jù)庫進(jìn)行操作,要對其它數(shù)據(jù)庫進(jìn)行操作,可以使用--database選項(xiàng),后面為數(shù)據(jù)庫的別名。
與此相應(yīng)的,dbshell,dumpdata,loaddata命令都有--database選項(xiàng)。
也可以手動(dòng)的選擇路由:
查詢:
>>> # This will run on the 'default' database.
>>> Author.objects.all()
>>> # So will this.
>>> Author.objects.using('default').all()
>>> # This will run on the 'other' database.
>>> Author.objects.using('other').all()
保存:
>>> my_object.save(using='legacy_users')
移動(dòng):
>>> p = Person(name='Fred') >>> p.save(using='first') # (statement 1) >>> p.save(using='second') # (statement 2)
以上的代碼會產(chǎn)生問題,當(dāng)p在first數(shù)據(jù)庫中第一次保存時(shí),會默認(rèn)生成一個(gè)主鍵,這樣使用second數(shù)據(jù)庫保存時(shí),p已經(jīng)有了主鍵,這個(gè)主鍵如果未被使用不會產(chǎn)生問題,但如果先前被使用了,就會覆蓋原先的數(shù)據(jù)。
有兩個(gè)解決方法;
1.保存前清除主鍵:
>>> p = Person(name='Fred') >>> p.save(using='first') >>> p.pk = None # Clear the primary key. >>> p.save(using='second') # Write a completely new object.
2.使用force_insert
>>> p = Person(name='Fred') >>> p.save(using='first') >>> p.save(using='second', force_insert=True)
刪除:
從哪個(gè)數(shù)據(jù)庫取得的對象,從哪刪除
>>> u = User.objects.using('legacy_users').get(username='fred')
>>> u.delete() # will delete from the `legacy_users` database
如果想把一個(gè)對象從legacy_users數(shù)據(jù)庫轉(zhuǎn)移到new_users數(shù)據(jù)庫:
>>> user_obj.save(using='new_users') >>> user_obj.delete(using='legacy_users')
以上就是本文的全部內(nèi)容,希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
- 詳解多線程Django程序耗盡數(shù)據(jù)庫連接的問題
- Django使用多數(shù)據(jù)庫的方法
- Django中數(shù)據(jù)庫的數(shù)據(jù)關(guān)系:一對一,一對多,多對多
- django 多數(shù)據(jù)庫配置教程
- Django app配置多個(gè)數(shù)據(jù)庫代碼實(shí)例
- Django多數(shù)據(jù)庫配置及逆向生成model教程
- django 多數(shù)據(jù)庫及分庫實(shí)現(xiàn)方式
- django 鏈接多個(gè)數(shù)據(jù)庫 并使用原生sql實(shí)現(xiàn)
- Django多數(shù)據(jù)庫聯(lián)用實(shí)現(xiàn)方法解析
- django使用多個(gè)數(shù)據(jù)庫的方法實(shí)例
相關(guān)文章
深入學(xué)習(xí)Python可變與不可變對象操作實(shí)例
Python中的數(shù)據(jù)類型可以分為可變對象和不可變對象,了解它們之間的區(qū)別對于編寫高效的Python代碼至關(guān)重要,本文將詳細(xì)介紹可變對象和不可變對象的概念,以及如何正確地使用它們來提高代碼的性能和可讀性2023-12-12
Python灰度變換中的分段線性函數(shù)專項(xiàng)分析實(shí)現(xiàn)
灰度變換是指根據(jù)某種目標(biāo)條件按一定變換關(guān)系逐點(diǎn)改變源圖像中每個(gè)像素灰度值的方法。目的是改善畫質(zhì),使圖像顯示效果更加清晰。圖像的灰度變換處理是圖像增強(qiáng)處理技術(shù)中的一種非?;A(chǔ)、直接的空間域圖像處理方法,也是圖像數(shù)字化軟件和圖像顯示軟件的一個(gè)重要組成部分2022-10-10
Python實(shí)現(xiàn)遞歸遍歷文件夾并刪除文件
本文給大家匯總了3個(gè)Python實(shí)現(xiàn)遍歷文件夾并刪除的代碼,主要是給大家分享下這3種方法的實(shí)現(xiàn)思路,有需要的小伙伴可以參考下2016-04-04
python獲取指定時(shí)間段內(nèi)特定規(guī)律的日期列表
這篇文章主要介紹了python獲取指定時(shí)間段內(nèi)特定規(guī)律的日期列表,本文通過實(shí)例代碼給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2021-04-04
初學(xué)python數(shù)學(xué)建模之?dāng)?shù)據(jù)導(dǎo)入(小白篇)
本篇文章是小白篇初學(xué)python的同學(xué)可以來共同學(xué)習(xí)了,本篇文章主要講解了python數(shù)學(xué)建模過程中的第一步數(shù)據(jù)導(dǎo)入,數(shù)據(jù)導(dǎo)入是所有數(shù)模編程的第一步,比你想象的更重要2021-08-08

