1. 程式人生 > 其它 >MySQL高階語言(二)

MySQL高階語言(二)

一、檢視

1、VIEW

1.1概念

  • 可以被當作是虛擬表或儲存查詢
  • 檢視跟表格的不同是,表格中有實際儲存資料,而檢視是建立在表格之上的一個架構,它本身並不實際儲存資料。
  • 臨時表在使用者退出或同資料庫的連線斷開後就自動消失了,而檢視不會消失。
  • 檢視不含有資料,只儲存它的定義,它的用途一般可以簡化複雜的查詢。比如你要對幾個表進行連線查詢,而且還要進行統計排序等操作,寫SQL語句會很麻煩的,用檢視將幾個表聯結起來,然後對這個檢視進行查詢操作,就和對一個表查詢一樣,很方便。

1.2建立、檢視和刪除檢視

CREATE VIEW "視圖表名" AS "SELECT 語句";                      #建立視圖表
SELECT 
* FROM `V_NAME_VALUE`; #檢視視圖表 DROP VIEW V_NAME_VALUE;        #刪除視圖表
例項操作:
mysql> create view v_test_01 as select A.name,A.address from test A where name in (select B.name from t1 B where score > 20);
mysql> select *from v_test_01;
drop view v_test_01;

2、聯集

將兩個SQL語句的結果合併起來,兩個SQL語句所產生的欄位需要是同樣的

2.1UNION

生成結果值將沒有重複,且按照欄位的順序進行排序
語法:[SELECT 語句 1] UNION [SELECT 語句 2];

例項操作:
select
name from t1 union select name from test;

2.2UNION ALL

將生成結果的值都列出來,無論有無重複
語法:[SELECT 語句 1] UNION ALL [SELECT 語句 2];

例項操作:
select name from t1 union all select name from test;

3、交集值(取兩個SQL語句結果的交集)

3.1取交集值的方法1

注:2種簡單方法,內連線+on/using,去重則加上distinct

select A.name from t1 A inner join test B on A.name=B.name;
select A.name from t1 A inner join test B using(name);
select distinct A.name from t1 A inner join test B on A.name=B.name;

3.2取交集方法2

1種,union all結合group by:兩表其中的一個表沒有指定的行,而另一個表這個行有重複不可用,要求兩個表確實有交集的時候用。

select A.name from (select name from t1 union all select name from test) A group by A.name having count(*) > 1;
select A.name from (select name from t1 union all select name from test) A group by A.name having count(name) > 1; 
select name from t1 union all select name from test; #拆分上面的SQL語句
select A.name,count(name) from (select name from t1 union all select name from test) A group by A.name having count(name) > 1; #顯示count值,便於理解
select A.name,count(name) from (select distinct name from t1 union all select distinct name from test) A group by A.name having count(name) > 1; #去重顯示,在聯集兩個表之前先把表去重,以防一個表中本身就有重複值

3.3取交集

取兩個SQL語句結果的交集,且沒有重複

方法一:
mysql> select A.name from (select B.name from t1 B inner join test C on B.name=C.name) A group by A.name;
方法二:
select distinct A.name from t1 A inner join test B using(name);
方法三:
select distinct name from t1 where name in (select name from test);
方法四:
select distinct A.name from t1 A left join test B using(name) where B.name is NOT NULL;

(1)內連線取交集結合group by去重

(2)內連線取交集結合distinct去重

(3)where+in遍歷取交集並結合distinct去重

(4)使用左連線(也可用右連線)+where 判斷NOT NULL 取交集並結合distinct去重

4、無交集值

顯示第一個SQL語句的結果,且與第二個SQL語句沒有交集的結果,且沒有重複

方法一:
select A.name from (select distinct name from t1 union all select distinct name from test) A group by A.name having count(name)=1;
方法二:
select distinct name from test where name not in (select distinct name from t1);
select distinct name from t1 where name not in (select distinct name from test);
方法三:
select distinct A.name from t1 A left join test B using(name) where B.name is NULL;
select distinct B.name from t1 A right join test B using(name) where A.name is NULL;

(1)union all結合group by進行分組彙總並使用count=1取無交集值

(2)where+not in遍歷取無交集值並結合distinct去重

(3)使用左連線(或者右連線)+where 判斷NULL 取無交集並結合distinct去重

5、CASE的用法

是SQL用來作為IF-THEN-ELSE之類邏輯的關鍵字

5.1語法格式

SELECT CASE (欄位名)
    WHEN "條件1" THEN  "結果1"
    WHEN "條件2" THEN  "結果2"
    ……
    ELSE "結果N"
    END
FROM "表名"

條件可以是一個數值或是公式。ELSE子句不是必須的

5.2例項操作

6、空值(NULL)和無值(“”)的區別

區別:
無值的長度為0,不佔用空間;而空值null 的長度是null,是佔用空間的;
IS NULL或者IS NOT NULL,是用來判斷欄位是不是NULL或者不是NULL,是不能查出是不是無值的;
無值的判斷使用=’‘或者<>’'來處理。<>代表不等於;
在通過count()指定欄位統計有多少行數時,如果遇到NULL值會自動忽略掉,遇到空值會自動加入記錄中進行計算。

6.1判斷空值和無值的字元長度

select length(NULL),length(''),length('1');

6.2使用count統計行數(體現null與空值的區別)

count(*) 表示包括所有列的行數,不會忽略null值;空值正常統計
count(列名) 表示只包括這一列,統計時會忽略null值的行;空值正常統計

二、正則表示式(REGEXP)

1、正則表示式匹配符 

2、語法

SELECT 選項 FROM 表名 WHERE 選項 REGEXP (模式)

select * from test where name regexp 'wu$';    ###以wu結尾的
select * from test where score regexp '50|80';    
select * from test1 where name regexp '^F|^Z';

三、儲存過程

1、概述

  • 儲存過程是一組為了完成特定功能的SQL語句集合
  • 儲存過程在使用過程中是將常用或者複雜的工作預先使用 SQL 語句寫好並用一個指定的名稱儲存起來,這個過程經編譯和優化後儲存在資料庫伺服器中,當需要使用時,只需要呼叫即可
  • 儲存過程在執行上比傳統SQL速度更快、執行效率更高。

2、優點

  • 執行一次後,會將生成的二進位制程式碼駐留緩衝區,提高執行效率
  • SQL語句加上控制語句的集合,靈活性高
  • 在伺服器端儲存,客戶端呼叫時,降低網路負載
  • 可多次重複被呼叫,可隨時修改,不影響客戶端呼叫
  • 可完成所有的資料庫操作,也可控制資料庫的資訊訪問許可權

3、建立、呼叫、檢視和刪除儲存過程

(1) 建立儲存過程

DELIMITER $$                                #將語句的結束符號從分號;臨時改為兩個$$ (可以是自定義)
CREATE PROCEDURE proc()                     #建立儲存過程,過程名為Proc, 不帶引數
-> BEGIN                                    #過程體以關鍵字BEGIN開始
-> SELECT * FROM test5;                     #過程體語句(自己根據需求進行編寫)
-> END $$                                   #過程體以關鍵字END結束
DELIMITER ;                                 #將語句的結束符號恢復為分號

實操:

mysql> delimiter ##
mysql> create procedure proc()
    -> begin
    -> insert into test2 values (2,'lucy',23);
    -> insert into test2 values (2,'jack',21);
    -> insert into test2 values (4,'nancy',22);
    -> select * from test2;
    -> end ##

  mysql> delimiter ;

(2)呼叫儲存過程

mysql> CALL proc;

(3)檢視儲存過程

SHOW CREATE PROCEDURE [資料庫.] 儲存過程名;      #檢視某個儲存過程的具體資訊(如果在指定庫中,庫名可以省略)
SHOW CREATE PROCEDURE fzr.proc;                 #未省略庫名
SHOW CREATE PROCEDURE proc;                     #省略庫名
 
SHOW PROCEDURE STATUS [LIKE '%proc%'] \G                #豎列檢視

(4)儲存過程的引數

  • IN輸入引數:表示呼叫者向過程傳入值(傳入值可以是字面量或變數)
  • OUT輸出引數:表示過程向呼叫者傳出值(可以返回多個值)(傳出值只能是變數)
  • INOUT輸入輸出引數:既表示呼叫者向過程傳入值,又表示過程向呼叫者傳出值(值只能是變數)

例項操作:
IN 傳入引數

mysql> delimiter ##
mysql> create procedure proc1(in iage int)
    -> begin
    -> select * from test2 where age > iage ;
    -> end ##
Query OK, 0 rows affected (0.00 sec)
 
mysql> delimiter ;
mysql> call proc1(21);

OUT輸出引數

因為out是向呼叫者輸出引數,不接收輸入的引數,所以儲存過程裡的num為null

#呼叫了proc2儲存過程,輸出引數,改變了num變數的值

INOUT輸入輸出引數

 呼叫了proc3儲存過程,接受了輸入的引數,也輸出引數,改變了變數

(5)刪除儲存過程

儲存過程內容的修改方法是通過刪除原有儲存過程之後以相同名稱建立新的儲存過程

DROP PROCEDURE IF EXISTS proc1;

4、儲存過程的控制語句

(1) 條件控制語句 if-then-else … end if

mysql> delimiter ##                                #修改預設結束符為##
mysql> create procedure proc1(in iage int)         #建立儲存過程proc1,引數為iage,資料型別為int
    -> begin                                       #過程體以關鍵詞begin開始
    -> declare var int;                            #定義變數var為int型別
    -> set var=iage*2;                             #設定變數var等於傳入引數的2倍
    -> if var >=20 then                            #如果var大於等於20,則執行下面的過程體
    -> update test2 set age=age+1;                 #設定test2中的age+1
    -> else                                        #如果變數var不大於10,則執行下面過程體
    -> update test2 set age=age-1;                 #設定表test2中的age-1
    -> end if;                                     #結束if語句
    -> end ##                                      #結束建立儲存過程
Query OK, 0 rows affected (0.00 sec)
 
mysql> delimiter ;                                 #重新修改預設結束符為原來的;
mysql> call proc1(10);                             #呼叫proc1的儲存過程,並傳入引數10
Query OK, 6 rows affected (0.00 sec)

(2)迴圈語句while … end while

mysql> delimiter ##                       #修改預設結束符為##
mysql> create procedure proc4()           #建立儲存過程為proc4
    -> begin                              #過程體以關鍵字begin開始
    -> declare var int;                   #定義變數var為int型別
    -> set var=0;                         #設定var=0
    -> while var <10 do                   #使用while迴圈,var要小於10
    -> insert into test2(id) values(var); #滿足條件則進行新增資料,內容為變數var
    -> set var=var+1;                     #變數var每次迴圈後加1
    -> end while;                         #結束while迴圈
    -> end ##                             #結束建立儲存過程
Query OK, 0 rows affected (0.00 sec)
 
mysql> delimiter ;                        #重新修改預設結束符為原來的;
mysql> call proc4;                        #呼叫proc4儲存過程
Query OK, 1 row affected (0.01 sec)