摘要:數(shù)據(jù)庫(kù)設(shè)計(jì)的三大范式確保每列的原子性如果每列都是不可再分的最小單元信息,則滿足第一范式,比如下圖中,地址是由國(guó)家和城市組成的,顯然可以繼續(xù)在拆分成兩個(gè)列,國(guó)家和城市,是不滿足第一范式的,需要將地址列差分成國(guó)家和城市兩個(gè)列。
閱讀原文 前言
這是關(guān)于 MySQL 系列文章的第三篇,在前兩篇文章 《MySQL —— 數(shù)據(jù)庫(kù)基礎(chǔ)》 和 《MySQL —— SQL 語(yǔ)句總結(jié)》 中,主要介紹了一些數(shù)據(jù)庫(kù)的基礎(chǔ)概念、創(chuàng)建表的方式以及 SQL 語(yǔ)句的使用,本篇在使用的基礎(chǔ)上做一個(gè)小小的升華,來(lái)簡(jiǎn)單聊一聊數(shù)據(jù)庫(kù)的設(shè)計(jì),還有一句話不得不再次贅述,數(shù)據(jù)庫(kù)博大精深,本系列文章內(nèi)容較淺,適合于前端的同學(xué)們對(duì) MySQL 的入門,這也是我的學(xué)習(xí)筆記,希望可以幫助大家。
為什么設(shè)計(jì)數(shù)據(jù)庫(kù)說(shuō)到為什么要設(shè)計(jì)數(shù)據(jù)庫(kù),就要說(shuō)到數(shù)據(jù)的完整性,我們要在設(shè)計(jì)數(shù)據(jù)庫(kù)時(shí)保證域的完整性和實(shí)體的完整性,同時(shí)從性能出發(fā),我們要保證最大限度的節(jié)省存儲(chǔ)空間,比如一張成績(jī)表,上面沒(méi)必要存儲(chǔ)學(xué)生的姓名、年齡等信息,只需要存儲(chǔ)成績(jī),如果一個(gè)數(shù)據(jù)庫(kù)設(shè)計(jì)的合理,最后的結(jié)果就是方便我們對(duì)數(shù)據(jù)庫(kù)的開發(fā)和擴(kuò)展。
如果是一個(gè) “糟糕” 的數(shù)據(jù)庫(kù)設(shè)計(jì)會(huì)造成一系列的不良反應(yīng),比如數(shù)據(jù)冗余,存儲(chǔ)空間浪費(fèi),內(nèi)存浪費(fèi),有時(shí)甚至?xí)斐蓴?shù)據(jù)插入和更新的異常,比如學(xué)生表存了學(xué)生信息,而成績(jī)表也存了,這樣在修改時(shí)沒(méi)有全部修改就會(huì)出現(xiàn)錯(cuò)誤。
軟件項(xiàng)目開發(fā)中數(shù)據(jù)庫(kù)設(shè)計(jì)的生命周期軟件項(xiàng)目開發(fā)中數(shù)據(jù)庫(kù)設(shè)計(jì)的生命周期可大概分為以下幾個(gè)階段:
需求分析階段,分析客戶的業(yè)務(wù)和數(shù)據(jù)處理需求;
概要設(shè)計(jì)階段,設(shè)計(jì)數(shù)據(jù)庫(kù) E-R 模型圖,確認(rèn)需求的正確和完整性;
詳細(xì)設(shè)計(jì)階段,應(yīng)用三大范式審核數(shù)據(jù)庫(kù);
代碼編寫階段,物理實(shí)現(xiàn)數(shù)據(jù)庫(kù),編碼實(shí)現(xiàn)應(yīng)用;
軟件測(cè)試階段;
安裝部署階段。
上面數(shù)據(jù)庫(kù)的設(shè)計(jì)經(jīng)歷了從 “現(xiàn)實(shí)世界” 到 “信息世界” 到 “數(shù)據(jù)庫(kù)模型” 再到 “數(shù)據(jù)庫(kù)” 產(chǎn)生的一個(gè)完整過(guò)程。
設(shè)計(jì)數(shù)據(jù)庫(kù)的步驟收集信息:收集信息其實(shí)就是與相關(guān)人員進(jìn)行交流、訪談、調(diào)研,充分了解用戶需求,理解整個(gè)項(xiàng)目的完整流程,并理解數(shù)據(jù)庫(kù)需要完成的任務(wù),這部分工作大部分由需求人員完成,并根技術(shù)人員進(jìn)行對(duì)接。
標(biāo)識(shí)實(shí)體和實(shí)體屬性:開發(fā)人員在明確需求和流程之后,標(biāo)識(shí)數(shù)據(jù)庫(kù)的實(shí)體,比如學(xué)生信息表,每一條實(shí)體中應(yīng)該由哪些字段組成,成績(jī)表中實(shí)體由哪些字段組成等等。
標(biāo)識(shí)實(shí)體之間的關(guān)系:其實(shí)就是通過(guò)表之間的某字段對(duì)表進(jìn)行關(guān)聯(lián),對(duì)表的實(shí)體之間建立對(duì)應(yīng)關(guān)系,如學(xué)生表的 id 字段會(huì)關(guān)聯(lián)成績(jī)表的 student_id 字段,用來(lái)查找某個(gè)學(xué)生的成績(jī)。
數(shù)據(jù)庫(kù) E-R 圖 1、E-R 圖基本概念E-R 圖也叫做實(shí)體關(guān)系圖,是指用實(shí)體、關(guān)系、屬性三個(gè)基本概念概括數(shù)據(jù)的基本結(jié)構(gòu),從而描述靜態(tài)數(shù)據(jù)的概念模型。
E-R 圖的實(shí)體:即數(shù)據(jù)模型中的數(shù)據(jù)對(duì)象,每一張表就是一個(gè) E-R 圖的實(shí)體。
E-R 圖的屬性:即數(shù)據(jù)對(duì)象中所具有的屬性,例如學(xué)生表的學(xué)生、姓名、年齡等,屬性又分為唯一屬性和非唯一屬性,唯一屬性如經(jīng)過(guò)唯一約束和主鍵約束的屬性,不可重復(fù),其他的都是非唯一屬性。
E-R 圖的關(guān)系:用來(lái)表示每一個(gè)數(shù)據(jù)對(duì)象與數(shù)據(jù)對(duì)象之間的聯(lián)系,即每一個(gè)實(shí)體之間的聯(lián)系,例如學(xué)生表和成績(jī)表之間的聯(lián)系,因?yàn)槊總€(gè)學(xué)生都有自己的成績(jī)。
2、E-R 圖的關(guān)聯(lián)關(guān)系 (1) 1 對(duì) 1 (1 : 1)1 對(duì) 1 關(guān)系是指對(duì)于實(shí)體集 A 和 實(shí)體集 B,A 中的每一個(gè)實(shí)體最多與 B 中的一個(gè)實(shí)體有關(guān)系,反之在實(shí)體集 B 中的每一個(gè)實(shí)體之多與實(shí)體集 A 中的一個(gè)實(shí)體有關(guān)系。
(2) 1 對(duì)多(1 : N)1 對(duì)多關(guān)系是指實(shí)體集 A 與實(shí)體集 B 中至少有 N (N > 0) 個(gè)實(shí)體有關(guān)系,并且實(shí)體集 B 中最多與實(shí)體集 A 中的一個(gè)實(shí)體有關(guān)系。
(3) 多對(duì)多(M : N)多對(duì)多關(guān)系是指實(shí)體集 A 中的每一個(gè)實(shí)體與實(shí)體集 B 中至少有 M (M > 0) 個(gè)實(shí)體有關(guān)系,并且實(shí)體集 B 中的每一個(gè)實(shí)體與實(shí)體集 A 中至少有 N (N > 0) 個(gè)實(shí)體有關(guān)系。
數(shù)據(jù)庫(kù)設(shè)計(jì)的三大范式 1、確保每列的原子性如果每列都是不可再分的最小單元信息,則滿足第一范式,比如下圖中,地址是由國(guó)家和城市組成的,顯然可以繼續(xù)在拆分成兩個(gè)列,國(guó)家和城市,是不滿足第一范式的,需要將地址列差分成國(guó)家和城市兩個(gè)列。
舉一個(gè)簡(jiǎn)單的例子,我們平時(shí)在淘寶購(gòu)物的時(shí)候需要添加地址,在填寫新地址時(shí),都是讓我們選擇國(guó)家、省、城市、區(qū)、街道、小區(qū)這樣的方式,而不是讓我們自己將這些地址寫在一起,其原因就是因?yàn)樘詫毜臄?shù)據(jù)庫(kù)設(shè)計(jì)嚴(yán)格遵循每列的原子性,這樣的提交可以方便后端獲取每一個(gè)列的信息在數(shù)據(jù)庫(kù)中進(jìn)行存儲(chǔ)。
2、每個(gè)表只能描述一件事情如下圖中所示,在左側(cè)的表中,描述了學(xué)生信息和課程信息,這明顯是兩件事情,假設(shè)再有一張成績(jī)表,也要描述學(xué)生信息,課程信息和成績(jī)等多件事情,就會(huì)造成數(shù)據(jù)的重復(fù)、冗余,也可能會(huì)導(dǎo)致更新、插入、刪除數(shù)據(jù)異常的現(xiàn)象。
所以正確的做法是應(yīng)該將左側(cè)表差分成兩張表分別為學(xué)生表和課程表,并使用學(xué)生編號(hào)與課程編號(hào)進(jìn)行關(guān)聯(lián)。
3、其他列都不傳遞依賴于主鍵列其他列都不傳遞依賴于主鍵列的意思是表中各列必須都與主鍵直接相關(guān),不能簡(jiǎn)介相關(guān),從下圖左表可以看出,學(xué)生編號(hào)為主鍵,年級(jí) ID 也應(yīng)該為主鍵,正常應(yīng)該通過(guò)學(xué)生編號(hào)找到年級(jí) ID,再找到年級(jí)名稱,這樣年級(jí)名稱與學(xué)生編號(hào)之間就形成了一個(gè)傳遞并且依賴于主鍵年級(jí) ID,即年級(jí) ID 做為主鍵在中間隔了一層,這樣就使年級(jí)名稱與主鍵學(xué)生編號(hào)間接相關(guān),如果在同一張表中,所有的字段都是應(yīng)該直接依賴于主鍵,而不是再通過(guò)其他的主鍵傳遞。
如果一個(gè)表中表述了多件事情并有多個(gè)作為主鍵的列,與上一條的處理方式相同,應(yīng)該拆成多張表,并且每張表只有一個(gè)主鍵列。
RBAC 基于角色的訪問(wèn)控制 1、RBAC 的含義RBAC(Role-Based Access Control)基于角色的訪問(wèn)控制,就是用戶通過(guò)角色與權(quán)限進(jìn)行關(guān)聯(lián),簡(jiǎn)單的說(shuō),一個(gè)用戶擁若干個(gè)角色,每個(gè)角色擁有若干個(gè)權(quán)限,這樣就構(gòu)造成了 “用戶 → 角色 → 權(quán)限 → 資源” 的授權(quán)模型,在這個(gè)模型中,用戶與角色之間,角色與權(quán)限之間,權(quán)限與資源之間,一般都是多對(duì)多的關(guān)系,在 RBAC 中最重要的概念主要有四部分,就是用戶(User)、角色(Role)、權(quán)限(Permission)和資源(Resource)。
2、RBAC 的安全原則最小權(quán)限原則:最小權(quán)限原則之所以被 RBAC 所支持,是因?yàn)?RBAC 可以將其角色配置成完成任務(wù)所需要的最小的權(quán)限集;
責(zé)任分離原則:可以通過(guò)調(diào)用相互獨(dú)立互斥的角色來(lái)共同完成敏感的任務(wù)而體現(xiàn),比如要求一個(gè)計(jì)帳員和財(cái)務(wù)管理員共參與同一個(gè)帳目;
數(shù)據(jù)抽象原則:數(shù)據(jù)抽象可以通過(guò)權(quán)限的抽象來(lái)體現(xiàn),如財(cái)務(wù)操作用借款、存款等抽象權(quán)限,而不用操作系統(tǒng)提供的典型的讀、寫、執(zhí)行權(quán)限。
3、RBAC 的 E-R 圖之前說(shuō) RBAC 最重要的概念由四部分,其實(shí)體現(xiàn)在數(shù)據(jù)庫(kù)的表中有主要三部分,因?yàn)榻巧陀脩羰侵丿B的,那么主要有三張表分別為用戶表、權(quán)限表和資源表,其中用戶表與權(quán)限表之間有一張關(guān)聯(lián)表,權(quán)限表與資源表之間有一張關(guān)聯(lián)表,E-R 圖如下。
事務(wù) 1、為什么需要事務(wù)?在生活中我們經(jīng)常使用銀行轉(zhuǎn)賬或者支付寶和微信支付,這種操作每一次至少影響兩個(gè)用戶的數(shù)據(jù)信息,比如一方給另一方轉(zhuǎn)錢,如果成功則轉(zhuǎn)錢方余額減去轉(zhuǎn)出金額,而收錢方余額增加收到的金額,這應(yīng)該是一個(gè)請(qǐng)求操作了數(shù)據(jù)表中的倆個(gè)實(shí)體,如果在兩個(gè)操作數(shù)據(jù)的環(huán)節(jié)任意一個(gè)失敗了,都會(huì)影響兩個(gè)人數(shù)據(jù)的正確性,這種時(shí)候需要兩個(gè)操作同時(shí)失敗或同時(shí)成功,就是說(shuō)有一個(gè)操作出現(xiàn)失敗的情況,即使另一個(gè)成功了也需要進(jìn)行回滾操作,這就是事務(wù)的由來(lái)。
2、什么是事務(wù)事務(wù)是作為單個(gè)邏輯工作單元執(zhí)行的一系列操作,多個(gè)操作作為一個(gè)整體向系統(tǒng)提交,要么都執(zhí)行,要么都不執(zhí)行,是一個(gè)不可分割的工作邏輯單元。
轉(zhuǎn)賬過(guò)程就是一個(gè)整體,它需要兩條 UPDATE 語(yǔ)句,如果任何一個(gè)出錯(cuò),則整個(gè)轉(zhuǎn)賬業(yè)務(wù)取消,兩個(gè)賬戶的余額都恢復(fù)到原來(lái)的數(shù)據(jù)(回滾),確保總余額不變。
這里再舉一個(gè)例子,有一個(gè)上傳文件的功能,后端接收到文件流時(shí)是需要先寫入的,當(dāng)寫入成功后,會(huì)將上傳成功的結(jié)果返回給客戶端,如果文件很大,寫入的時(shí)間就會(huì)長(zhǎng),如果在此期間突然寫入失敗,則會(huì)刪除之前寫入的內(nèi)容,將整個(gè)操作回滾到寫入之前,這里面主要兩步操作,創(chuàng)建一個(gè)新文件并寫入,寫入成功刪除舊文件,如果寫入失敗,兩個(gè)操作將會(huì)同時(shí)失敗,即不會(huì)刪除舊文件,這也是一個(gè)事務(wù)的例子,只是沒(méi)有轉(zhuǎn)賬那么明顯。
3、事務(wù)的特性 ACID事務(wù)具有以下特性,被簡(jiǎn)稱為 ACID:
原子性(Atomicity):事務(wù)是一個(gè)完整的操作,事務(wù)各個(gè)部分是不可分的,要么都執(zhí)行,要么都不執(zhí)行;
一致性(Consistency):當(dāng)事務(wù)完成后,數(shù)據(jù)必須處理完整的狀態(tài);
隔離性(Isolation):并發(fā)事務(wù)彼此隔離、獨(dú)立,它不應(yīng)該以任何方式依賴于其它事務(wù);
持久性(Durability):事務(wù)完成后,它對(duì)數(shù)據(jù)庫(kù)的修改被永久保持。
4、如何創(chuàng)建事務(wù) (1) 創(chuàng)建表創(chuàng)建表 account,id 列為主鍵列,name 列為姓名,balance 為余額。
CREATE TABLE `account` ( `id` INT(11) NOT NULL AUTO_INCREMENT, `name` VARCHAR(64) NOT NULL, `balance` INT(11) DEFAULT 0 PRIMARY KEY (`id`) );(2) 添加數(shù)據(jù)
將表 account 添加兩條數(shù)據(jù),分別為 “張三” 和 “李四”,余額都為 100。
INSERT INTO `student` (`name`, `balance`) VALUES ("張三", 100); INSERT INTO `student` (`name`, `balance`) VALUES ("李四", 100);(3) 使用 NodeJS 實(shí)現(xiàn)事務(wù)
const mysql = require("mysql"); // 創(chuàng)建數(shù)據(jù)庫(kù)連接 const connection = mysql.createConnection({ host: "localhost", // 主機(jī)名 port: "3306", // 數(shù)據(jù)庫(kù)服務(wù)端口號(hào) username: "root", // 數(shù)據(jù)庫(kù)名稱 pwd: "123456", // 數(shù)據(jù)庫(kù)密碼 database: "school" // 連接的數(shù)據(jù)庫(kù)名稱 }); connection.connect(); // 開啟事務(wù) connection.beginTransaction(err => { // 回調(diào)參數(shù)為錯(cuò)誤對(duì)象,返回結(jié)果,返回字段描述 connection.query("UPDATE account SET balance - 50 WHERE id = 1", (err, result, fields) => { if (err) { connection.rollback(); // 如果失敗直接回歸 } else { connection.query("UPDATE account SET balance - 50 WHERE id = 1", (err, result, fields) => { if (err) { connection.rollback(); // 如果失敗直接回歸 } else { connection.commit(); // 如果兩個(gè)都成功了則提交事務(wù) } }); } }); });總結(jié)
到此關(guān)于 MySQL 的系列文章就告一段落了,希望前端的同學(xué)們?cè)诳戳诉@幾篇文章后對(duì)你們?nèi)腴T MySQL 有一些幫助,那這幾篇的文章就達(dá)到目的了,也歡迎后端的小伙伴來(lái)指出文章中的錯(cuò)誤和不足。
文章版權(quán)歸作者所有,未經(jīng)允許請(qǐng)勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉(zhuǎn)載請(qǐng)注明本文地址:http://systransis.cn/yun/17791.html
摘要:編輯器編輯器背景編輯器前段時(shí)間遇到一個(gè)線上問(wèn)題,后來(lái)排查好久發(fā)現(xiàn)是因?yàn)橹鲝耐窖舆t導(dǎo)致的,所以今天寫一篇文章總結(jié)一下這個(gè)問(wèn)題希望對(duì)你有用。編輯器幾句嘮叨編輯器大家好,我是小飯,一枚后端工程師。背景前段時(shí)間遇到一個(gè)線上問(wèn)題,后來(lái)排查好久發(fā)現(xiàn)是因?yàn)橹鲝耐窖舆t導(dǎo)致的,所以今天寫一篇文章總結(jié)一下這個(gè)問(wèn)題希望對(duì)你有用。如果覺(jué)得還不錯(cuò),記得加個(gè)關(guān)注點(diǎn)個(gè)贊哦思維導(dǎo)圖思維導(dǎo)圖常見(jiàn)的主從架構(gòu)隨著日益增長(zhǎng)的訪...
摘要:要快,但是我們的服務(wù)也必須萬(wàn)無(wú)一失,后續(xù)我會(huì)分享百度移動(dòng)端首頁(yè)的前端架構(gòu)設(shè)計(jì)那么這樣的優(yōu)化,是如何做到的呢,又如何兼顧穩(wěn)定性,架構(gòu)性,與速度呢別急,讓我們把這些優(yōu)化一一道來(lái)。百度移動(dòng)端首頁(yè)的很多就是這樣緩存在客戶端的。 歡迎大家收看聊一聊系列,這一套系列文章,可以幫助前端工程師們了解前端的方方面面(不僅僅是代碼):https://segmentfault.com/blog/fronte...
摘要:前言最近在使用框架,看了下他的源碼,發(fā)現(xiàn)有很多地方也用到了依賴注入控制反轉(zhuǎn),覺(jué)得有必要和大家簡(jiǎn)單聊一聊什么是依賴注入以及怎么使用它。概念依賴注入和控制反轉(zhuǎn)是對(duì)同一件事情的不同描述,從某個(gè)方面講,就是它們描述的角度不同。 前言 最近在使用ThinkPHP5框架,看了下他的源碼,發(fā)現(xiàn)有很多地方也用到了依賴注入(控制反轉(zhuǎn)),覺(jué)得有必要和大家簡(jiǎn)單聊一聊什么是依賴注入以及怎么使用它。 簡(jiǎn)介 I...
閱讀 2138·2021-09-27 14:04
閱讀 1883·2019-08-30 15:55
閱讀 1707·2019-08-30 13:13
閱讀 1075·2019-08-30 13:07
閱讀 2752·2019-08-29 15:20
閱讀 3247·2019-08-29 12:42
閱讀 3344·2019-08-28 17:58
閱讀 3605·2019-08-28 17:56