返回顶部

Django基础之读写分离

一. 简介

前提:MySQL数据库已经配置好了主从复制,且至少有两个数据库,然后在django中进行读写分离的操作。

 

下面的例子就用sqlite3来说明:

 

1.1 基本配置

第一步:首先配置多个数据库,在settings配置文件中配置以下内容:

DATABASES = {
    'default': { #默认数据库,配置多个mysql数据也是ok的,混用数据库也是ok的
        'ENGINE': 'django.db.backends.sqlite3',
        'NAME': os.path.join(BASE_DIR, 'db.sqlite3'),
    },
    'db2': { #配置的第二个数据库,注意数据库名字不能相同
        'ENGINE': 'django.db.backends.sqlite3',
        'NAME': os.path.join(BASE_DIR, 'db2.sqlite3'),
    }
}

 

第二步:执行数据库同步指令,将我们的表结构生成到db2这个数据库中。如果是已经有了主从复制的数据库,就不要执行这些操作,直接配置连接即可。

python migrate --database db2(库名)

 

第三步:视图中的简单使用

def dbtest(request):
    data = []
    # 向db2数据库中写入数据
    # models.Class.objects.using('db2').create(name='xx')

    # 从db1中读取数据
    data = models.Class.objects.using('default').all()
    # 更新时
    for i in data:
        i.name = 'xxxxxx'
        i.save(using='default') #更新时指定数据库
    return render(request,'dbtest.html',{'data':data})

  配置自动选择:

  在应用文件夹中创建一个py文件,名字随意,比如叫做router.py文件,写上下面的内容:

class Router:
    # 读操作用default库,就return这个库名字符串
    def db_for_read(self,model,**kwargs):
        return 'default'

    def db_for_write(self,model,**kwargs):
        return 'db2'

然后在settings配置文件中做下面的配置

DATABASE_ROUTERS = ['app01.router.Router',] #写上面这个类的路径,我的是在app01应用文件夹下面的router.py文件中了

那么在逻辑的views.py文件中写的时候,就不需要using指定数据库了:

def dbtest(request):
    data = []
    # 向db2数据库中写入数据
    models.Class.objects.create(name='xx2')

    # 从db1中读取数据
    data = models.Class.objects.all()

    return render(request,'dbtest.html',{'data':data})

 

1.2 一主多从的情况

import random
class Router:
    # 读操作用default库,就return这个库名字符串
    def db_for_read(self,model,**kwargs):
        print(11111)
        print(model._meta.app_label)
        return 'default'
        # return random.choice(['db1','db2','db3']) #多个库读的时候,可以简单的写个随机选择

    def db_for_write(self,model,**kwargs):

        return 'db2'

根据不同的应用来选择不同的库进行读取

 
import random
class Router:
    # 读操作用default库,就return这个库名字符串
    def db_for_read(self,model,**kwargs):
        print(model)
        print(dir(model))# 其中有个_meta属性很有用
        a = model._meta.app_label #获取当前model对象所在的应用名称
     m = model._meta.model_name 获取当前操作的model对象的表名,也可以根据表名来进行多数据库读的分配
# 可以根据应用选择不用的库来进行读取 if a == 'app01': return 'db1' elif a == 'app02': return 'db2' return 'default' def db_for_write(self,model,**kwargs):

 

posted @ 2020-10-28 22:51  muguangrui  阅读(252)  评论(0编辑  收藏  举报