主頁 > 後端開發 > django使用多個資料庫實作

django使用多個資料庫實作

2023-04-22 07:27:52 後端開發

一、說明:

  在開發 Django 專案的時候,很多時候都是使用一個資料庫,即 settings 中只有 default 資料庫,但是有一些專案確實也需要使用多個資料庫,這樣的專案,在資料庫配置和使用的時候,就比較麻煩一點,

二、Django使用多個資料庫中settings中的DATABASES的設定

  2.1 默認只是用一個資料庫時 DATABASES 的設定(以 SQLite 為例)

DATABASES = {
    'default': {
        'ENGINE': 'django.db.backends.sqlite3',         
        'NAME': 'db.sqlite3',
    }
}

  2.2 Django 資料庫支持的 ENGINE 型別

    • 'django.db.backends.postgresql'
    • 'django.db.backends.mysql'
    • 'django.db.backends.sqlite3'
    • 'django.db.backends.oracle'

  2.3 設定了多個資料庫后 settings 中的 DATABASES 的設定

DATABASES = {
    'default': {
        'ENGINE': 'django.db.backends.sqlite3',
        'NAME': 'db.sqlite3',
    },
    'db1': {
        'ENGINE': 'django.db.backends.mysql',
        'NAME': 'mysql_test_db1',
        'USER': 'root',
        'PASSWORD': 'Se7eN521',
        'HOST': '127.0.0.1',
        'PORT': '3306'
    },
    'db2': {
        'ENGINE': 'django.db.backends.mysql',
        'NAME': 'mysql_test_db2',
        'USER': 'root',
        'PASSWORD': 'Se7eN521',
        'HOST': '127.0.0.1',
        'PORT': '3306'
    }
}

三、實作思路

  1. 多個應用對應多個資料庫和一個應用對應多個資料庫
    1. 情況一:專案有多個 應用app 且需要使用到多個資料庫
    2. 情況二:專案只有一個應用app, 且但需要使用到多個資料庫,
  2. 這兩種情況的實作思路其實都是一樣的,都是為每個資料庫創建一個應用,即這個應用只對接一個資料庫,如果這個應用不需要寫任何業務邏輯的代碼,也需要創建一個空的應用,主要是用來做資料庫遷移的
  3. 核心思想就是:一個model類對應一個資料庫,通過資料庫路由和model定義時指定的all_label來實作,

四、案例實作

  第一步:創建需要的 應用app,并且在 INSTALLED_APPS 中參考

    其中db1_app這個應用主要是用來對接資料庫db1的

    其中db2_app這個應用主要是用來對接資料庫db2的

    其中test_app這個應用主要用來實作業務邏輯的

                 

  第二步:創建 應用app 和 資料庫之間的映射關系

    在settings.py 檔案夾中設定 DATABASE_APPS_MAPPING 的字典,里面主要是配置 應用app 和資料庫的對應關系

DATABASE_APPS_MAPPING = {
    "db1_app": "db1",   # db1_app 對應 db1 資料庫
    "db2_app": "db2"    # db2_app 對應 db2 資料庫
}

  第三步:創建資料庫路由

    在專案的主檔案夾即 settings.py 的同目錄下創建一個 database_router.py 檔案,該檔案的作用就是給不同應用app 配置不同的資料庫,

# _*_ coding:utf-8 _*_
# @Time : 2023/4/20 5:37 下午

from django.conf import settings

DATABASE_MAPPING = settings.DATABASE_APPS_MAPPING
print('DATABASE_MAPPING = {}'.format(DATABASE_MAPPING))

class DatabaseAppsRouter(object):

    # 設定 應用app 讀取時資料庫的設定
    def db_for_read(self, model, **hints)if model._meta.app_label in DATABASE_MAPPING:
            return DATABASE_MAPPING[model._meta.app_label]
        return None

    def db_for_write(self, model, **hints):
        if model._meta.app_label in DATABASE_MAPPING:
            return DATABASE_MAPPING[model._meta.app_label]
        return None

    def allow_relation(self, obj1, obj2, **hints):
        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_migrate(self, db, app_label, model_name=None, **hints):
        """
        Make sure that apps only appear in the related database.
        根據app_label的值只在相應的資料庫中創建一個表,如果洗掉該def或
        不指定過濾條件,則一個Model會在每個資料庫里都創建一個表,
        """
        if db in DATABASE_MAPPING.values():return DATABASE_MAPPING.get(app_label) == db
        elif app_label in DATABASE_MAPPING:
            return False
        return None

  第四步:在setting.py中配置 DATABASE_ROUTERS 指定自由路由檔案:

#test_django為專案名,database_router為路由檔案名,DatabaseAppsRouter為路由中創建的類名
DATABASE_ROUTERS = ['django_db_demo.database_router.DatabaseAppsRouter']

  第五步:創建model類

    說明:model 可以根據需要卸載任何一個應用app的model.py檔案中,也可以分散寫在多個應用的model.py中,這個根據自己的需要即可,但是如何推薦一定要在model類的Meta中指定app_label,不然會全部將表創建到default資料庫中

from django.db import models

class SqliteModel(models.Model):
    """帳號和用戶關聯"""

    sqlite_name = models.CharField(max_length=20)
    class Meta:
        # 當前這個 SqliteModel 定義的資料庫的表將會創建在test_app 對應的default 資料庫中
        app_label = "test_app"      # 當有多個資料庫鏈接的時候,要通過app_label 來區分這個model對應那個資料庫
   
class Db1Model(models.Model):
    """帳號和用戶關聯"""

    db1_name = models.CharField(max_length=20)
    class Meta:
        # 當前這個Db1Model 定義的資料庫的表將會創建在 db1_app 對應的 db1 資料庫中
        app_label = "db1_app"        # 當有多個資料庫鏈接的時候,要通過app_label 來區分這個model對應那個資料庫
       
class Db2Model(models.Model):
    """帳號和用戶關聯"""

    db2_name = models.CharField(max_length=20)
    class Meta:
        # 當前這個Db2Model 定義的資料庫的表將會創建在 db2_app 對應的 db1 資料庫中
        app_label = "db2_app"       # 當有多個資料庫鏈接的時候,要通過app_label 來區分這個model對應那個資料庫
      

  

  第六步:資料遷移

python3 manage.py makemigrations  
python3 manage.py migrate --database=default   # 當有多個資料庫,需要遷移多次
python3 manage.py migrate --database=db1
python3 manage.py migrate --database=db2 

  第七步:查看遷移:

    model對應的表,分別遷移到不同的資料庫成功,剩下的增刪改查的就正常引入model物件即可,這樣就實作了,不同的model物件,對應不用資料庫的表,

    

第五步:總結

  1. 創建多個資料庫連接設定
  2. 創建多個資料與應用app的映射關系
  3. 創建資料庫路由
  4. 創建model類的時候置指明app_label,即這個model是屬于那個app,從而覺得遷移到那個資料庫

侯哥語錄:我曾經是一個職業教育者,現在是一個自由開發者,我希望我的分享可以和更多人一起進步,分享一段我喜歡的話給大家:"我所理解的自由不是想干什么就干什么,而是想不干什么就不干什么,當你還沒有能力說不得時候,就努力讓自己變得強大,擁有說不得權利,"

轉載請註明出處,本文鏈接:https://www.uj5u.com/houduan/550758.html

標籤:Python

上一篇:【K哥爬蟲普法】微信公眾號爬蟲構成不正當競爭,爬蟲er面對金山,如何避免濫用爬蟲?

下一篇:返回列表

標籤雲
其他(157775) Python(38085) JavaScript(25379) Java(17985) C(15215) 區塊鏈(8256) C#(7972) AI(7469) 爪哇(7425) MySQL(7135) html(6777) 基礎類(6313) sql(6102) 熊猫(6058) PHP(5869) 数组(5741) R(5409) Linux(5327) 反应(5209) 腳本語言(PerlPython)(5129) 非技術區(4971) Android(4555) 数据框(4311) css(4259) 节点.js(4032) C語言(3288) json(3245) 列表(3129) 扑(3119) C++語言(3117) 安卓(2998) 打字稿(2995) VBA(2789) Java相關(2746) 疑難問題(2699) 细绳(2522) 單片機工控(2479) iOS(2429) ASP.NET(2402) MongoDB(2323) 麻木的(2285) 正则表达式(2254) 字典(2211) 循环(2198) 迅速(2185) 擅长(2169) 镖(2155) 功能(1967) .NET技术(1959) Web開發(1951) python-3.x(1918) HtmlCss(1917) 弹簧靴(1913) C++(1910) xml(1889) PostgreSQL(1872) .NETCore(1854) 谷歌表格(1846) Unity3D(1843) for循环(1842)

熱門瀏覽
  • 【C++】Microsoft C++、C 和匯編程式檔案

    ......

    uj5u.com 2020-09-10 00:57:23 more
  • 例外宣告

    相比于斷言適用于排除邏輯上不可能存在的狀態,例外通常是用于邏輯上可能發生的錯誤。 例外宣告 Item 1:當函式不可能拋出例外或不能接受拋出例外時,使用noexcept 理由 如果不打算拋出例外的話,程式就會認為無法處理這種錯誤,并且應當盡早終止,如此可以有效地阻止例外的傳播與擴散。 示例 //不可 ......

    uj5u.com 2020-09-10 00:57:27 more
  • Codeforces 1400E Clear the Multiset(貪心 + 分治)

    鏈接:https://codeforces.com/problemset/problem/1400/E 來源:Codeforces 思路:給你一個陣列,現在你可以進行兩種操作,操作1:將一段沒有 0 的區間進行減一的操作,操作2:將 i 位置上的元素歸零。最終問:將這個陣列的全部元素歸零后操作的最少 ......

    uj5u.com 2020-09-10 00:57:30 more
  • UVA11610 【Reverse Prime】

    本人看到此題沒有翻譯,就附帶了一個自己的翻譯版本 思考 這一題,它的第一個要求是找出所有 $7$ 位反向質數及其質因數的個數。 我們應該需要質數篩篩選1~$10^{7}$的所有數,這里就不慢慢介紹了。但是,重讀題,我們突然發現反向質數都是 $7$ 位,而將它反過來后的數字卻是 $6$ 位數,這就說明 ......

    uj5u.com 2020-09-10 00:57:36 more
  • 統計區間素數數量

    1 #pragma GCC optimize(2) 2 #include <bits/stdc++.h> 3 using namespace std; 4 bool isprime[1000000010]; 5 vector<int> prime; 6 inline int getlist(int ......

    uj5u.com 2020-09-10 00:57:47 more
  • C/C++編程筆記:C++中的 const 變數詳解,教你正確認識const用法

    1、C中的const 1、區域const變數存放在堆疊區中,會分配記憶體(也就是說可以通過地址間接修改變數的值)。測驗代碼如下: 運行結果: 2、全域const變數存放在只讀資料段(不能通過地址修改,會發生寫入錯誤), 默認為外部聯編,可以給其他源檔案使用(需要用extern關鍵字修飾) 運行結果: ......

    uj5u.com 2020-09-10 00:58:04 more
  • 【C++犯錯記錄】VS2019 MFC添加資源不懂如何修改資源宏ID

    1. 首先在資源視圖中,添加資源 2. 點擊新添加的資源,復制自動生成的ID 3. 在解決方案資源管理器中找到Resource.h檔案,編輯,使用整個專案搜索和替換的方式快速替換 宏宣告 4. Ctrl+Shift+F 全域搜索,點擊查找全部,然后逐個替換 5. 為什么使用搜索替換而不使用屬性視窗直 ......

    uj5u.com 2020-09-10 00:59:11 more
  • 【C++犯錯記錄】VS2019 MFC不懂的批量添加資源

    1. 打開資源頭檔案Resource.h,在其中預先定義好宏 ID(不清楚其實ID值應該設定多少,可以先新建一個相同的資源項,再在這個資源的ID值的基礎上遞增即可) 2. 在資源視圖中選中專案資源,按F7編輯資源檔案,按 ID 型別 相對路徑的形式添加 資源。(別忘了先把檔案拷貝到專案中的res檔案 ......

    uj5u.com 2020-09-10 01:00:19 more
  • C/C++編程筆記:關于C++的參考型別,專供新手入門使用

    今天要講的是C++中我最喜歡的一個用法——參考,也叫別名。 參考就是給一個變數名取一個變數名,方便我們間接地使用這個變數。我們可以給一個變數創建N個參考,這N + 1個變數共享了同一塊記憶體區域。(參考型別的變數會占用記憶體空間,占用的記憶體空間的大小和指標型別的大小是相同的。雖然參考是一個物件的別名,但 ......

    uj5u.com 2020-09-10 01:00:22 more
  • 【C/C++編程筆記】從頭開始學習C ++:初學者完整指南

    眾所周知,C ++的學習曲線陡峭,但是花時間學習這種語言將為您的職業帶來奇跡,并使您與其他開發人員區分開。您會更輕松地學習新語言,形成真正的解決問題的技能,并在編程的基礎上打下堅實的基礎。 C ++將幫助您養成良好的編程習慣(即清晰一致的編碼風格,在撰寫代碼時注釋代碼,并限制類內部的可見性),并且由 ......

    uj5u.com 2020-09-10 01:00:41 more
最新发布
  • django使用多個資料庫實作

    一、說明: 在開發 Django 專案的時候,很多時候都是使用一個資料庫,即 settings 中只有 default 資料庫,但是有一些專案確實也需要使用多個資料庫,這樣的專案,在資料庫配置和使用的時候,就比較麻煩一點。 二、Django使用多個資料庫中settings中的DATABASES的設定 ......

    uj5u.com 2023-04-22 07:27:52 more
  • 【K哥爬蟲普法】微信公眾號爬蟲構成不正當競爭,爬蟲er面對金山,如

    我國目前并未出臺專門針對網路爬蟲技術的法律規范,但在司法實踐中,相關判決已屢見不鮮,K 哥特設了“K哥爬蟲普法”專欄,本欄目通過對真實案例的分析,旨在提高廣大爬蟲工程師的法律意識,知曉如何合法合規利用爬蟲技術,警鐘長鳴,做一個守法、護法、有原則的技術人員。 案情介紹 2011年1月微信問世,騰訊公司 ......

    uj5u.com 2023-04-22 07:22:31 more
  • LocalDateTime

    // LocalDateTime類: 獲取日期時間資訊。格式為 2018-09-06T15:33:56.750 // 得到指定日期時間 LocalDateTime dateTime = LocalDateTime.of(1985, 4, 15, 12, 12, 12); // 得到當前日期時間 Lo ......

    uj5u.com 2023-04-22 07:18:40 more
  • 網路流的C++代碼實作與程序講解

    網路流是一種非常重要的圖論演算法,它在許多實際問題中得到廣泛應用。本文將介紹網路流演算法的C++代碼實作與程序講解。 演算法概述 網路流演算法是通過將圖中的邊看作流量通道,將圖的點看作流量的起點或終點,來求解圖中的最大或最小流量的問題。它是一種非常重要的最優化演算法,廣泛應用于圖論、運籌學、計算機網路等領域。 ......

    uj5u.com 2023-04-21 09:15:02 more
  • java反射的一些理解

    首先簡單介紹下java反射的應用場景:java反射多用于框架設計中。 其次,簡述下框架:可重復用的,用來提高編程效率的代碼。一些重復性的作業不需要在去開發,直接利用框架集成起來,用的時候呼叫框架,傳遞引數等等。 再次,介紹下java反射獲取類物件的三種方式: Class.forName("全限定類名 ......

    uj5u.com 2023-04-21 09:14:58 more
  • java反射的一些理解

    首先簡單介紹下java反射的應用場景:java反射多用于框架設計中。 其次,簡述下框架:可重復用的,用來提高編程效率的代碼。一些重復性的作業不需要在去開發,直接利用框架集成起來,用的時候呼叫框架,傳遞引數等等。 再次,介紹下java反射獲取類物件的三種方式: Class.forName("全限定類名 ......

    uj5u.com 2023-04-21 09:13:56 more
  • 網路流的C++代碼實作與程序講解

    網路流是一種非常重要的圖論演算法,它在許多實際問題中得到廣泛應用。本文將介紹網路流演算法的C++代碼實作與程序講解。 演算法概述 網路流演算法是通過將圖中的邊看作流量通道,將圖的點看作流量的起點或終點,來求解圖中的最大或最小流量的問題。它是一種非常重要的最優化演算法,廣泛應用于圖論、運籌學、計算機網路等領域。 ......

    uj5u.com 2023-04-21 09:13:08 more
  • Opencv在VS2022中的配置(Python)

    下載Opencv 先去官網https://opencv.org/opencv-4-7-0/下載, 找到適合你設備的版本下載Windows就是Win pack,完成后進行安裝即可,一路同意默認就行,可以更改安裝位置,但路徑上盡可能以英文,以防止后面不必要的問題。 2.下載Python 首先是版本 發文 ......

    uj5u.com 2023-04-21 07:42:17 more
  • 【Visual Leak Detector】原始碼下載

    說明 使用 VLD 記憶體泄漏檢測工具輔助開發時整理的學習筆記。本篇介紹 VLD 原始碼的下載。同系列文章目錄可見 《記憶體泄漏檢測工具》目錄 1. 下載途徑 以 v2.5.1 版本為例,可以到 Github-KindDragon-vld 頁面下載 master 的 zip 原始碼包,如下所示: 也可以到 ......

    uj5u.com 2023-04-21 07:42:02 more
  • 原來這就是所謂的 JSR!

    相信大家在學習 Java 的程序中,或多或少都見過 JSR 這個詞。本篇文章就科普下什么是 JSR。 什么是 JSR ? JSR(Java Specification Requests),是指 Java 規范請求(或者活規范提案)。這個請求(提案)是提給 JCP 的(Java Community P ......

    uj5u.com 2023-04-21 07:41:54 more