<em>Mac</em>Book项目 2009年学校开始实施<em>Mac</em>Book项目,所有师生配备一本<em>Mac</em>Book,并同步更新了校园无线网络。学校每周进行电脑技术更新,每月发送技术支持资料,极大改变了教学及学习方式。因此2011
2021-06-01 09:32:01
最近在公司售前售後同事遇到一些奇怪的需求找到我,需要提供公司一些專案資料庫所有表的結構資訊(欄位名、型別、長度、是否主鍵、***、備註),雖然不是本職工作,但是作為python技能的擁有者看到這種需求還是覺得很容易的,但是如果不用程式碼解決確實非常棘手和浪費時間。於是寫了一個輕量小型專案來解決一些燃眉之急,希望能對一些人有所幫助,程式碼大神、小神可以忽略此貼。
想要匯出mysql資料庫表結構必須瞭解一些相關資料庫知識,mysql資料庫支援通過SQL語句進行表資訊查詢:
查詢資料庫所有表名
SHOW TABLES
查詢對應資料庫對應表結構資訊
SELECT COLUMN_NAME,COLUMN_TYPE,COLUMN_KEY,IS_NULLABLE, COLUMN_COMMENT FROM information_schema.`COLUMNS` WHERE TABLE_SCHEMA='{dbName}' AND TABLE_NAME='{tableName}'
以下是一個較為通用的mysql資料庫連線類,建立 MysqlConnection 類,放入對應資料庫連線資訊即可使用sql,通過query查詢、update增刪改、close關閉連線。
*注:資料量過大時不推薦直接使用query查詢。
import pymysql class MysqlConnection(): def __init__(self, host, user, passw, port, database, charset="utf8"): self.db = pymysql.connect(host=host, user=user, password=passw, port=port, database=database, charset=charset) self.cursor = self.db.cursor() # 查 def query(self, sql): self.cursor.execute(sql) results = self.cursor.fetchall() return results # 增刪改 def update(self, sql): try: self.cursor.execute(sql) self.db.commit() return 1 except Exception as e: print(e) self.db.rollback() return 0 # 關閉連線 def close(self): self.cursor.close() self.db.close()
config.yml,這裡使用了組態檔進行程式引數設定,方便設定一鍵執行
# 資料庫資訊設定 db_config: host: 127.0.0.1 # 資料庫所在服務IP port: 3306 # 資料庫伺服器埠 username: root # ~使用者名稱 password: 12346 # ~密碼 charset: utf8 # 需要進行處理的資料名稱列表 《《 填入資料庫名 db_names: ['db_a','db_b'] # 匯出設定 excel_conf: # 匯出結構Excel表頭,長度及順序不可調整,僅支援更換名稱 column_name: ['欄位名', '資料型別', '長度', '主鍵', '非空', '描述'] save_dir: ./data
讀取組態檔的程式碼
import yaml class Configure(): def __init__(self): with open("config.yaml", 'r', encoding='utf-8') as f: self._conf = yaml.load(f.read(), Loader=yaml.FullLoader) def get_db_config(self): host = self._conf['db_config']['host'] port = self._conf['db_config']['port'] username = self._conf['db_config']['username'] password = self._conf['db_config']['password'] charset = self._conf['db_config']['charset'] db_names = self._conf['db_config']['db_names'] return host, port, username, password, charset, db_names def get_excel_title(self): title = self._conf['excel_conf']['column_name'] save_dir = self._conf['excel_conf']['save_dir'] return title, save_dir
利用上面建立的資料庫連線和SQL查詢獲取所有表
class ExportMysqlTableStructureInfoToExcel(): def __init__(self): conf = Configure() # 獲取設定初始化類資訊 self.__host, self.__port, self.__username, self.__password, self.__charset, self.db_names = conf.get_db_config() self.__excel_title, self.__save_dir = conf.get_excel_title() ```省略``` def __connect_to_mysql(self, database): # 獲取資料庫連線方法 connect = MysqlConnection(self.__host, self.__username, self.__password, self.__port, database, self.__charset) return connect def __get_all_tables(self, con): # 查詢所有表 res = con.query("SHOW TABLES") tb_list = [] for item in res: tb_list.append(item[0]) return tb_list ``````
迴圈獲取每一張表的結構資料,根據需要對中英文做了一些轉換,欄位長度可以從型別中分離出來,這裡使用yield返回資料,可以利用生成器加速處理過程(外包匯出儲存和資料庫查詢可以並行)
class ExportMysqlTableStructureInfoToExcel(): ```省略``` def __struct_of_table_generator(self, con, db_name): tb_list = self.__get_all_tables(con) for index, tb_name in enumerate(tb_list): sql = "SELECT COLUMN_NAME,COLUMN_TYPE,COLUMN_KEY,IS_NULLABLE, COLUMN_COMMENT " "FROM information_schema.`COLUMNS` WHERE TABLE_SCHEMA='{}' AND TABLE_NAME='{}'".format(db_name, tb_name) res = con.query(sql) struct_list = [] for item in res: column_name, column_type, column_key, is_nullable, column_comment = item length = "0" if str(column_type).find('(') > -1: column_type, length = str(column_type).replace(")", '').split('(') if column_key == 'PRI': column_key = "是" else: column_key = '' if is_nullable == 'YES': is_nullable = '是' else: is_nullable = '否' struct_list.append([column_name, column_type, length, column_key, is_nullable, column_comment]) yield [struct_list, tb_name] ```省略```
class ExportMysqlTableStructureInfoToExcel(): ```省略``` def export(self): if len(self.db_names) == 0: print("請設定資料庫列表") for i, db_name in enumerate(self.db_names): # 對多個資料庫進行處理 connect = self.__connect_to_mysql(db_name) # 獲取資料庫連線 if not os.path.exists(self.__save_dir): # 判斷資料匯出儲存路徑是否存在 os.mkdir(self.__save_dir) file_name = os.path.join(self.__save_dir,'{}.xlsx'.format(db_name)) # 用資料庫名命名匯出Excel檔案 if not os.path.exists(file_name): # 檔案不存在時自動建立檔案 excel wrokb = openpyxl.Workbook() wrokb.save(file_name) wrokb.close() wb = openpyxl.load_workbook(file_name) writer = pd.ExcelWriter(file_name, engine='openpyxl') writer.book = wb struct_generator = self.__struct_of_table_generator(connect, db_name) # 獲取表結構資訊的生成器 for tb_info in tqdm(struct_generator, desc=db_name): # 從生成器中獲取表結構並利用pandas進行格式化儲存,寫入Excel檔案 s_list, tb_name = tb_info data = pd.DataFrame(s_list, columns=self.__excel_title) data.to_excel(writer, sheet_name=tb_name) writer.close() connect.close() ```省略```
由於資料表太多,手動編寫耗費的時間太久,所以搞了一個簡單的指令碼快速生成資料庫結構,儲存到word檔案中。
1.安裝pymysql和document
pip install pymysql pip install document
2.指令碼
# -*- coding: utf-8 -*- import pymysql from docx import Document from docx.shared import Pt from docx.oxml.ns import qn db = pymysql.connect(host='127.0.0.1', #資料庫伺服器IP port=3306, user='root', passwd='123456', db='test_db') #資料庫名稱) #根據表名查詢對應的欄位相關資訊 def query(tableName): #開啟資料庫連線 cur = db.cursor() sql = "select b.COLUMN_NAME,b.COLUMN_TYPE,b.COLUMN_COMMENT from (select * from information_schema.`TABLES` where TABLE_SCHEMA='test_db') a right join(select * from information_schema.`COLUMNS` where TABLE_SCHEMA='test_db_test') b on a.TABLE_NAME = b.TABLE_NAME where a.TABLE_NAME='" + tableName+"'" cur.execute(sql) data = cur.fetchall() cur.close return data #查詢當前庫下面所有的表名,表名:tableName;表名+註釋(用於填充至word檔案):concat(TABLE_NAME,'(',TABLE_COMMENT,')') def queryTableName(): cur = db.cursor() sql = "select TABLE_NAME,concat(TABLE_NAME,'(',TABLE_COMMENT,')') from information_schema.`TABLES` where TABLE_SCHEMA='test_db_test'" cur.execute(sql) data = cur.fetchall() return data #將每個表生成word結構,輸出到word檔案 def generateWord(singleTableData,document,tableName): p=document.add_paragraph() p.paragraph_format.line_spacing=1.5 #設定該段落 行間距為 1.5倍 p.paragraph_format.space_after=Pt(0) #設定段落 段後 0 磅 #document.add_paragraph(tableName,style='ListBullet') r=p.add_run('n'+tableName) r.font.name=u'宋體' r.font.size=Pt(12) table = document.add_table(rows=len(singleTableData)+1, cols=3,style='Table Grid') table.style.font.size=Pt(11) table.style.font.name=u'Calibri' #設定表頭樣式 #這裡只生成了三個表頭,可通過實際需求進行修改 for i in ((0,'NAME'),(1,'TYPE'),(2,'COMMENT')): run = table.cell(0,i[0]).paragraphs[0].add_run(i[1]) run.font.name = 'Calibri' run.font.size = Pt(11) r = run._element r.rPr.rFonts.set(qn('w:eastAsia'), '宋體') for i in range(len(singleTableData)): #設定表格內資料的樣式 for j in range(len(singleTableData[i])): run = table.cell(i+1,j).paragraphs[0].add_run(singleTableData[i][j]) run.font.name = 'Calibri' run.font.size = Pt(11) r = run._element r.rPr.rFonts.set(qn('w:eastAsia'), '宋體') #table.cell(i+1, 0).text=singleTableData[i][1] #table.cell(i+1, 1).text=singleTableData[i][2] #table.cell(i+1, 2).text=singleTableData[i][3] if __name__ == '__main__': #定義一個document document = Document() #設定字型預設樣式 document.styles['Normal'].font.name = u'宋體' document.styles['Normal']._element.rPr.rFonts.set(qn('w:eastAsia'), u'宋體') #獲取當前庫下所有的表名資訊和表註釋資訊 tableList = queryTableName() #迴圈查詢資料庫,獲取表欄位詳細資訊,並呼叫generateWord,生成word資料 #由於時間匆忙,我這邊選擇的是直接查詢資料庫,執行了100多次查詢,可以進行優化,查詢出所有的表結構,在程式碼裡面將每個表結構進行拆分 for singleTableName in tableList: data = query(singleTableName[0]) generateWord(data,document,singleTableName[1]) #儲存至檔案 document.save('資料庫設計.docx')
3.生成的word檔案預覽
執行成功後會在目錄下的data資料夾中看到儲存的Excel檔案(以資料庫名為單位儲存成檔案),每個Excel第一個tab是空的(一個小bug暫未解決),其他每個tab以對應表名進行命名。
程式碼很簡單,供各位學習參考。
到此這篇關於利用Python批次匯出mysql資料庫表結構的文章就介紹到這了,更多相關Python批次匯出mysql表結構內容請搜尋it145.com以前的文章或繼續瀏覽下面的相關文章希望大家以後多多支援it145.com!
相關文章
<em>Mac</em>Book项目 2009年学校开始实施<em>Mac</em>Book项目,所有师生配备一本<em>Mac</em>Book,并同步更新了校园无线网络。学校每周进行电脑技术更新,每月发送技术支持资料,极大改变了教学及学习方式。因此2011
2021-06-01 09:32:01
综合看Anker超能充系列的性价比很高,并且与不仅和iPhone12/苹果<em>Mac</em>Book很配,而且适合多设备充电需求的日常使用或差旅场景,不管是安卓还是Switch同样也能用得上它,希望这次分享能给准备购入充电器的小伙伴们有所
2021-06-01 09:31:42
除了L4WUDU与吴亦凡已经多次共事,成为了明面上的厂牌成员,吴亦凡还曾带领20XXCLUB全队参加2020年的一场音乐节,这也是20XXCLUB首次全员合照,王嗣尧Turbo、陈彦希Regi、<em>Mac</em> Ova Seas、林渝植等人全部出场。然而让
2021-06-01 09:31:34
目前应用IPFS的机构:1 谷歌<em>浏览器</em>支持IPFS分布式协议 2 万维网 (历史档案博物馆)数据库 3 火狐<em>浏览器</em>支持 IPFS分布式协议 4 EOS 等数字货币数据存储 5 美国国会图书馆,历史资料永久保存在 IPFS 6 加
2021-06-01 09:31:24
开拓者的车机是兼容苹果和<em>安卓</em>,虽然我不怎么用,但确实兼顾了我家人的很多需求:副驾的门板还配有解锁开关,有的时候老婆开车,下车的时候偶尔会忘记解锁,我在副驾驶可以自己开门:第二排设计很好,不仅配置了一个很大的
2021-06-01 09:30:48
不仅是<em>安卓</em>手机,苹果手机的降价力度也是前所未有了,iPhone12也“跳水价”了,发布价是6799元,如今已经跌至5308元,降价幅度超过1400元,最新定价确认了。iPhone12是苹果首款5G手机,同时也是全球首款5nm芯片的智能机,它
2021-06-01 09:30:45