面試提升及平常學習積累(二)——數據庫篇

這一篇主要講mysql的一些積累和學習體會。html

一:數據庫基礎知識

Innodb引擎

Innodb引擎提供了對數據庫ACID事務的支持,而且實現了SQL標準的四種隔離級別,關於數據庫事務與其隔離級別的內容請見數據庫事務與其隔離級別這篇文章。該引擎還提供了行級鎖和外鍵約束,它的設計目標是處理大容量數據庫系統,它自己其實就是基於MySQL後臺的完整數據庫系統,MySQL運行時Innodb會在內存中創建緩衝池,用於緩衝數據和索引。可是該引擎不支持FULLTEXT類型的索引,並且它沒有保存表的行數,當SELECT COUNT(*) FROM TABLE時須要掃描全表。當須要使用數據庫事務時,該引擎固然是首選。因爲鎖的粒度更小,寫操做不會鎖定全表,因此在併發較高時,使用Innodb引擎會提高效率。可是使用行級鎖也不是絕對的,若是在執行一個SQL語句時MySQL不能肯定要掃描的範圍,InnoDB表一樣會鎖全表。java

MyIASM引擎

MyIASM是MySQL默認的引擎,可是它沒有提供對數據庫事務的支持,也不支持行級鎖和外鍵,所以當INSERT(插入)或UPDATE(更新)數據時即寫操做須要鎖定整個表,效率便會低一些。不過和Innodb不一樣,MyIASM中存儲了表的行數,因而SELECT COUNT(*) FROM TABLE時只須要直接讀取已經保存好的值而不須要進行全表掃描。若是表的讀操做遠遠多於寫操做且不須要數據庫事務的支持,那麼MyIASM也是很好的選擇。mysql

惟一索引(彙集索引)

單個表只能有一個惟一索引,索引列的值必須惟一,但容許有空值。若是是組合索引,則列值的組合必須惟一。數據存儲結構和索引存儲結構順序一致。(通常主鍵爲惟一索引)面試

建立單列索引:CREATE UNIQUE  INDEX indexName ON tableName(tableColumns(length));若是是CHAR,VARCHAR類型,length能夠小於字段實際長度;若是是BLOB 和 TEXT 類型,必須指定length。sql

建立組合索引:CREATE UNIQUE  INDEX indexName ON tableName( `column1`, `column2`, `column3` );數據庫

刪除索引:DROP INDEX indexName;網絡

非彙集索引

單個表能夠有多個非彙集索引,能夠是單列也能夠是組合索引,數據存儲結構和索引存儲結構順序無關。架構

建立單列索引:CREATE INDEX indexName ON tableName(tableColumns(length));若是是CHAR,VARCHAR類型,length能夠小於字段實際長度;若是是BLOB 和 TEXT 類型,必須指定length。併發

建立組合索引:CREATE INDEX indexName ON tableName( `column1`, `column2`, `column3` );數據庫設計

刪除索引:DROP INDEX indexName;

組合索引

建立組合索引:CREATE INDEX indexName ON tableName( `column1`, `column2`, `column3` );

對於複合索引:Mysql從左到右的使用索引中的字段,一個查詢能夠只使用索引中的一部份,但只能是最左側部分。

例如索引是key index (a,b,c). 能夠支持a | a,b| a,b,c 3種組合進行查找,但不支持 b|c|b,c進行查找 ,當最左側字段是常量引用時,索引就十分有效。(組合索引最多2-3列)

視圖與存儲過程

這兩塊的內容能夠更具具體的業務須要進行創建,通常對於查詢較多的類似組合的表,或者須要隱藏原表信息的能夠合併提供查詢視圖,對於複雜業務和須要特殊效率的業務能夠提供存儲過程。

經常使用函數

-------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------

如下內容來自請去這裏查看更詳細  http://www.jb51.net/article/40179.htm 

--------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------

一、聚合函數(經常使用於GROUP BY從句的SELECT查詢中)
AVG(col)返回指定列的平均值
COUNT(col)返回指定列中非NULL值的個數
MIN(col)返回指定列的最小值
MAX(col)返回指定列的最大值
SUM(col)返回指定列的全部值之和
GROUP_CONCAT(col) 返回由屬於一組的列值鏈接組合而成的結果

二、字符串函數
CONCAT(s1,s2...,sn)將s1,s2...,sn鏈接成字符串
CONCAT_WS(sep,s1,s2...,sn)將s1,s2...,sn鏈接成字符串,並用sep字符間隔
LCASE(str)或LOWER(str) 返回將字符串str中全部字符改變爲小寫後的結果
REVERSE(str) 返回顛倒字符串str的結果

三、日期和時間函數
CURDATE()或CURRENT_DATE() 返回當前的日期
CURTIME()或CURRENT_TIME() 返回當前的時間
DATE_ADD(date,INTERVAL int keyword)返回日期date加上間隔時間int的結果(int必須按照關鍵字進行格式化),如:SELECTDATE_ADD(CURRENT_DATE,INTERVAL 6 MONTH);
DATE_FORMAT(date,fmt)  依照指定的fmt格式格式化日期date值
DATE_SUB(date,INTERVAL int keyword)返回日期date加上間隔時間int的結果(int必須按照關鍵字進行格式化),如:SELECTDATE_SUB(CURRENT_DATE,INTERVAL 6 MONTH);
DAYOFWEEK(date)   返回date所表明的一星期中的第幾天(1~7)
DAYOFMONTH(date)  返回date是一個月的第幾天(1~31)
DAYOFYEAR(date)   返回date是一年的第幾天(1~366)
DAYNAME(date)   返回date的星期名,如:SELECT DAYNAME(CURRENT_DATE);
FROM_UNIXTIME(ts,fmt)  根據指定的fmt格式,格式化UNIX時間戳ts
HOUR(time)   返回time的小時值(0~23)
MINUTE(time)   返回time的分鐘值(0~59)
MONTH(date)   返回date的月份值(1~12)
MONTHNAME(date)   返回date的月份名,如:SELECT MONTHNAME(CURRENT_DATE);
NOW()    返回當前的日期和時間
QUARTER(date)   返回date在一年中的季度(1~4),如SELECT QUARTER(CURRENT_DATE);
WEEK(date)   返回日期date爲一年中第幾周(0~53)
YEAR(date)   返回日期date的年份(1000~9999)

四、控制流函數
MySQL有4個函數是用來進行條件操做的,這些函數能夠實現SQL的條件邏輯,容許開發者將一些應用程序業務邏輯轉換到數據庫後臺。

MySQL控制流函數:
CASE WHEN[test1] THEN [result1]...ELSE [default] END若是testN是真,則返回resultN,不然返回default
CASE [test] WHEN[val1] THEN [result]...ELSE [default]END  若是test和valN相等,則返回resultN,不然返回default
IF(test,t,f)   若是test是真,返回t;不然返回f
IFNULL(arg1,arg2) 若是arg1不是空,返回arg1,不然返回arg2
NULLIF(arg1,arg2) 若是arg1=arg2返回NULL;不然返回arg1
這些函數的第一個是IFNULL(),它有兩個參數,而且對第一個參數進行判斷。若是第一個參數不是NULL,函數就會向調用者返回第一個參數;若是是NULL,將返回第二個參數。

代碼以下:

SELECT CASE 'green'
     WHEN 'red' THEN 'stop'
     WHEN 'green' THEN 'go' END;
SELECT CASE 9 WHEN 1 THEN 'a' WHEN 2 THEN 'b' ELSE 'N/A' END;
SELECT CASE WHEN (2+2)=4 THEN 'OK' WHEN(2+2)<>4 THEN 'not OK' END ASSTATUS;
SELECT Name,IF((IsActive = 1),'已激活','未激活') AS RESULT FROMUserLoginInfo;
SELECT fname,lname,(math+sci+lit) AS total,
CASE WHEN (math+sci+lit) < 50 THEN 'D'
WHEN (math+sci+lit) BETWEEN 50 AND 150 THEN 'C'
WHEN (math+sci+lit) BETWEEN 151 AND 250 THEN 'B'
ELSE 'A' END
AS grade FROM marks;
SELECT IF(ENCRYPT('sue','ts')=upass,'allow','deny') AS LoginResultFROM users WHERE uname = 'sue';#一個登錄驗證

-----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------

二:數據庫設計

在官方學習的時候這部分是要符合三範式規則的,可是這種規則下對於大多數的數據業務來講都是在理論上太完整,不太實用,因此能夠學習這種設計的思想,可是適不適用能夠按照本身的業務規劃。

在設計數據庫的過程主要能夠參考如下幾個方向(我的總結):

1.首先分析本身的業務,抽象好數據對象以及對象之間的關聯關係。

2.對於通用的一些數據字段能夠抽象總結出來,好比建立時間,更新時間,操做者。。。這些通用的字段。

3.針對本身的業務主要分析查詢的部分,對於重複查詢較多,修改機會較少的字段能夠適當的在多個表中進行冗餘,以提升查詢和簡化sql代碼的複雜度。

4.在建表時能夠考慮後期業務的擴展性,可是沒必要強制流出字段等,由於業務變化比較平凡,過多的預留會讓表看起來很是混亂。

5.當業務擴展性和項目需求有衝突時能夠優先考慮項目需求,至於擴展性可看業務發展再定,甚至修改。(我的觀點,可不理會)

6.作好字段說明,備註。

三:sql優化(只是優化的方向和原則)

1.級聯查詢優化,多個單標查詢的優化,能夠考慮用多表級聯查詢優化。

2.考慮索引使用狀況優化,對於破壞索引,未使用索引的sql,能夠考慮創建索引或者使現有索引生效方面優化。

3.對於大數據量分頁的優化。

4.經過分析業務,用更多的條件肯定更少的列。

 四:數據庫架構

1.對於大數據和有讀寫壓力的業務能夠考慮從幾個方向去設計   

表分區http://www.cnblogs.com/freeton/p/4265228.html  這個網址寫的不錯,值得瀏覽下。

分表存儲:按業務需求能夠有水平分表和垂直分表兩種不一樣的技術。

        水平分表是按照必定的規則將一個大的表格水平分割到不一樣的子表中,因此每一個表格的列都是相同的。

     垂直分表是將一個大的表格按照不一樣的業務需求把不一樣的列分割到不一樣的子表中,以某一個相同的屬性(例ID)關聯起來。

     對於分表後的查詢能夠按照業務先進行邏輯判斷,而後定位到某個子表。對於一些統計功能,則循環全部子表,合併。

分庫技術:就是採用不一樣的數據庫部署數據庫集羣,能夠搜搜MyCat中間件。

讀寫分離http://blog.csdn.net/zoomyj/article/details/50610349  這片文章寫了好多 有興趣的能夠多瞭解下,做爲一個瞭解和積累。

 

總結:零零散散寫了一些,這裏寫的主要是一些方向性的東西,可能沒有具體的一些實現或者是代碼。因此只是一個指引,但願能幫助你們在這些方面去學習,去積累。也有一些東西是網絡摘抄的,只要是好的就學習吧。

以上內適用一到三年java開發面試,我的看法,值得參考,以爲適用的順手推薦下,讓更多的人看到

最後分享一個不錯的雞湯給想要幹事的朋友們,祝你們成功!http://www.iheima.com/zixun/2017/0628/163820.shtml

相關文章
相關標籤/搜索