Django中使用多数据库的两种方式
Django项目开发中,ORM默认的连接对对象对象都是settings.DATABASES['default']
如果一个Django项目中有多个应用,分别需要使用不同的数据库,该如何配置呢?
第一步:在settings中配置多个数据库对象:
DATABASES = {
'default': {},
'common': {
# 'ENGINE': 'django.db.backends.sqlite3',
# 'NAME': os.path.join(BASE_DIR, 'db.sqlite3'),
'ENGINE': 'django.db.backends.mysql',
'HOST': '127.0.0.1',
'PORT': 3306,
'NAME': 'common',
'USER': 'root',
'PASSWORD': '123456.com'
},
'bug_trace': {
# 'ENGINE': 'django.db.backends.sqlite3',
# 'NAME': os.path.join(BASE_DIR, 'db.sqlite3'),
'ENGINE': 'django.db.backends.mysql',
'HOST': '127.0.0.1',
'PORT': 3306,
'NAME': 'bug_trace',
'USER': 'root',
'PASSWORD': '123456.com'
},
'student_v2': {
# 'ENGINE': 'django.db.backends.sqlite3',
# 'NAME': os.path.join(BASE_DIR, 'db.sqlite3'),
'ENGINE': 'django.db.backends.mysql',
'HOST': '127.0.0.1',
'PORT': 3306,
'NAME': 'student_v2',
'USER': 'root',
'PASSWORD': '123456.com'
}
}
- 在这里我定义了4个数据库,分别是:
- default,空数据库,即使用不到,也需要定义一个空对象。
- common,公用数据库,存放项目公用的信息(用户,session等)。
- bug_trace,代码管理的应用数据库。
- student_v2,学生管理的应用数据库。
方法一:使用数据库路由
第二步:在settings中配置多个数据库路由:
DATABASE_ROUTERS = ['common.database_router.DatabaseAppsRouter']
DATABASE_APPS_MAPPING = {
'auth': 'common', # django内置数据库
'admin': 'common', # django内置数据库
'contenttypes': 'common', # django内置数据库
'sessions': 'common', # django内置数据库
'messages': 'common', # django内置数据库
'staticfiles': 'common', # django内置数据库
'bug_trace': 'bug_trace',
'student_v2': 'student_v2',
}
- DATABASE_ROUTERS,表示路由对象:
- common.database_router.DatabaseAppsRouter代表common目录下database_router.py文件中的DatabaseAppsRouter类。
- DATABASE_APPS_MAPPING表示应用与数据库的映射关系,需要一一对应,我这里将一些公用组件全部对应到common数据库中。
第三步:在对应的目录下创建类文件:
# -*- coding:utf-8 -*-
from django.conf import settings
DATABASE_MAPPING = settings.DATABASE_APPS_MAPPING
class DatabaseAppsRouter(object):
def db_for_read(self, model, **hints):
""""Point all read operations to the specific database."""
if model._meta.app_label in DATABASE_MAPPING:
return DATABASE_MAPPING[model._meta.app_label]
return None
def db_for_write(self, model, **hints):
"""Point all write operations to the specific database."""
if model._meta.app_label in DATABASE_MAPPING:
return DATABASE_MAPPING[model._meta.app_label]
return None
def allow_relation(self, obj1, obj2, **hints):
"""Allow any relation between apps that use the same database."""
db_obj1 = DATABASE_MAPPING.get(obj1._meta.app_label)
db_obj2 = DATABASE_MAPPING.get(obj2._meta.app_label)
if db_obj1 and db_obj2:
if db_obj1 == db_obj2:
return True
else:
return False
return None
def allow_syncdb(self, db, model):
"""Make sure that apps only appear in the related database."""
if db in DATABASE_MAPPING.values():
return DATABASE_MAPPING.get(model._meta.app_label) == db
elif model._meta.app_label in DATABASE_MAPPING:
return False
return None
def allow_migrate(self, db, app_label, model_name=None, **hints):
db, app_label, model_name, hints
if db in DATABASE_MAPPING.values():
return DATABASE_MAPPING.get(app_label) == db
elif app_label in DATABASE_MAPPING:
return False
return None
第四步:数据库迁移:
python3 manage.py makemigrations
# 仅迁移common数据库对应的应用
python3 manage.py migrate --database='common'
# 仅迁移student_v2数据库对应的应用
python3 manage.py migrate --database='student_v2'
# 仅迁移bug_trace2数据库对应的应用
python3 manage.py migrate --database='bug_trace'
第五步:正常使用ORM语句:
from django.http import HttpResponse, HttpResponseRedirect
from student_v2.models import Student
# 查询接口
def student_retrieve(request, action=None, sno=None):
# 获取全部用户
student = list(Student.objects.all().values())
return HttpResponse(str(student))
方法二:手动指定数据库:
这种方式适合于不需要进行数据迁移的应用,即意味着通过运维直接建表、修改表结构等,而不是通过代码完成。
第二步:使用ORM语句时使用using函数:
from django.http import HttpResponse, HttpResponseRedirect
from student_v2.models import Student
# 查询接口
def student_retrieve(request, action=None, sno=None):
# 获取全部用户
student = list(Student.objects.using('student_v2').all().values())
return HttpResponse(str(student))
- using('student_v2'),指定使用student_v2数据库进行查询。