主頁 > 後端開發 > mysql高級day1

mysql高級day1

2020-09-15 04:11:04 後端開發

Mysql高級-day01

MySQL高級課程簡介

序號 Day01 Day02 Day03 Day04
1 Linux系統安裝MySQL 體系結構 應用優化 MySQL 常用工具
2 索引 存盤引擎 查詢快取優化 MySQL 日志
3 視圖 優化SQL步驟 記憶體管理及優化 MySQL 主從復制
4 存盤程序和函式 索引使用 MySQL鎖問題 綜合案例
5 觸發器 SQL優化 常用SQL技巧

1. Linux 系統安裝MySQL

1.1 下載Linux 安裝包

https://dev.mysql.com/downloads/mysql/5.7.html#downloads

1.2 安裝MySQL

1). 卸載 centos 中預安裝的 mysql
	
	rpm -qa | grep -i mysql
	
	rpm -e mysql-libs-5.1.71-1.el6.x86_64 --nodeps
	
2). 上傳 mysql 的安裝包
	
	alt + p -------> put  E:/test/MySQL-5.6.22-1.el6.i686.rpm-bundle.tar

3). 解壓 mysql 的安裝包 
	
	mkdir mysql
	
	tar -xvf MySQL-5.6.22-1.el6.i686.rpm-bundle.tar -C /root/mysql
	
4). 安裝依賴包 
	
	yum -y install libaio.so.1 libgcc_s.so.1 libstdc++.so.6 libncurses.so.5 --setopt=protected_multilib=false
	
	yum  update libstdc++-4.4.7-4.el6.x86_64
	
5). 安裝 mysql-client
	
	rpm -ivh MySQL-client-5.6.22-1.el6.i686.rpm

6). 安裝 mysql-server
	
	rpm -ivh MySQL-server-5.6.22-1.el6.i686.rpm
	

1.3 啟動 MySQL 服務

service mysql start

service mysql stop

service mysql status

service mysql restart

1.4 登錄MySQL

mysql 安裝完成之后, 會自動生成一個隨機的密碼, 并且保存在一個密碼檔案中 : /root/.mysql_secret

mysql -u root -p 

登錄之后, 修改密碼 :

set password = password('itcast');

授權遠程訪問 : 

grant all privileges on *.* to 'root' @'%' identified by 'itcast';
flush privileges;

2. 索引

2.1 索引概述

MySQL官方對索引的定義為:索引(index)是幫助MySQL高效獲取資料的資料結構(有序),在資料之外,資料庫系統還維護者滿足特定查找演算法的資料結構,這些資料結構以某種方式參考(指向)資料, 這樣就可以在這些資料結構上實作高級查找演算法,這種資料結構就是索引,如下面的示意圖所示 :

左邊是資料表,一共有兩列七條記錄,最左邊的是資料記錄的物理地址(注意邏輯上相鄰的記錄在磁盤上也并不是一定物理相鄰的),為了加快Col2的查找,可以維護一個右邊所示的二叉查找樹,每個節點分別包含索引鍵值和一個指向對應資料記錄物理地址的指標,這樣就可以運用二叉查找快速獲取到相應資料,

一般來說索引本身也很大,不可能全部存盤在記憶體中,因此索引往往以索引檔案的形式存盤在磁盤上,索引是資料庫中用來提高性能的最常用的工具,

2.2 索引優勢劣勢

優勢

1) 類似于書籍的目錄索引,提高資料檢索的效率,降低資料庫的IO成本,

2) 通過索引列對資料進行排序,降低資料排序的成本,降低CPU的消耗,

劣勢

1) 實際上索引也是一張表,該表中保存了主鍵與索引欄位,并指向物體類的記錄,所以索引列也是要占用空間的,

2) 雖然索引大大提高了查詢效率,同時卻也降低更新表的速度,如對表進行INSERT、UPDATE、DELETE,因為更新表時,MySQL 不僅要保存資料,還要保存一下索引檔案每次更新添加了索引列的欄位,都會調整因為更新所帶來的鍵值變化后的索引資訊,

2.3 索引結構

索引是在MySQL的存盤引擎層中實作的,而不是在服務器層實作的,所以每種存盤引擎的索引都不一定完全相同,也不是所有的存盤引擎都支持所有的索引型別的,MySQL目前提供了以下4種索引:

  • BTREE 索引 : 最常見的索引型別,大部分索引都支持 B 樹索引,
  • HASH 索引:只有Memory引擎支持 , 使用場景簡單 ,
  • R-tree 索引(空間索引):空間索引是MyISAM引擎的一個特殊索引型別,主要用于地理空間資料型別,通常使用較少,不做特別介紹,
  • Full-text (全文索引) :全文索引也是MyISAM的一個特殊索引型別,主要用于全文索引,InnoDB從Mysql5.6版本開始支持全文索引,

我們平常所說的索引,如果沒有特別指明,都是指B+樹(多路搜索樹,并不一定是二叉的)結構組織的索引,其中聚集索引、復合索引、前綴索引、唯一索引默認都是使用 B+tree 索引,統稱為 索引,

2.3.1 BTREE 結構

BTree又叫多路平衡搜索樹,一顆m叉的BTree特性如下:

  • 樹中每個節點最多包含m個孩子,
  • 除根節點與葉子節點外,每個節點至少有[ceil(m/2)]個孩子,
  • 若根節點不是葉子節點,則至少有兩個孩子,
  • 所有的葉子節點都在同一層,
  • 每個非葉子節點由n個key與n+1個指標組成,其中[ceil(m/2)-1] <= n <= m-1

以5叉BTree為例,key的數量:公式推導[ceil(m/2)-1] <= n <= m-1,所以 2 <= n <=4 ,當n>4時,中間節點分裂到父節點,兩邊節點分裂,

插入 C N G A H E K Q M F W L T Z D P R X Y S 資料為例,

演變程序如下:

1). 插入前4個字母 C N G A

2). 插入H,n>4,中間元素G字母向上分裂到新的節點

3). 插入E,K,Q不需要分裂

4). 插入M,中間元素M字母向上分裂到父節點G

5). 插入F,W,L,T不需要分裂

6). 插入Z,中間元素T向上分裂到父節點中

7). 插入D,中間元素D向上分裂到父節點中,然后插入P,R,X,Y不需要分裂

8). 最后插入S,NPQR節點n>5,中間節點Q向上分裂,但分裂后父節點DGMT的n>5,中間節點M向上分裂

到此,該BTREE樹就已經構建完成了, BTREE樹 和 二叉樹 相比, 查詢資料的效率更高, 因為對于相同的資料量來說,BTREE的層級結構比二叉樹小,因此搜索速度快,

2.3.3 B+TREE 結構

B+Tree為BTree的變種,B+Tree與BTree的區別為:

1). n叉B+Tree最多含有n個key,而BTree最多含有n-1個key,

2). B+Tree的葉子節點保存所有的key資訊,依key大小順序排列,

3). 所有的非葉子節點都可以看作是key的索引部分,

由于B+Tree只有葉子節點保存key資訊,查詢任何key都要從root走到葉子,所以B+Tree的查詢效率更加穩定,

2.3.3 MySQL中的B+Tree

MySql索引資料結構對經典的B+Tree進行了優化,在原B+Tree的基礎上,增加一個指向相鄰葉子節點的鏈表指標,就形成了帶有順序指標的B+Tree,提高區間訪問的性能,

MySQL中的 B+Tree 索引結構示意圖:

2.4 索引分類

1) 單值索引 :即一個索引只包含單個列,一個表可以有多個單列索引

2) 唯一索引 :索引列的值必須唯一,但允許有空值

3) 復合索引 :即一個索引包含多個列

2.5 索引語法

索引在創建表的時候,可以同時創建, 也可以隨時增加新的索引,

準備環境:

create database demo_01 default charset=utf8mb4;

use demo_01;

CREATE TABLE `city` (
  `city_id` int(11) NOT NULL AUTO_INCREMENT,
  `city_name` varchar(50) NOT NULL,
  `country_id` int(11) NOT NULL,
  PRIMARY KEY (`city_id`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8;

CREATE TABLE `country` (
  `country_id` int(11) NOT NULL AUTO_INCREMENT,
  `country_name` varchar(100) NOT NULL,
  PRIMARY KEY (`country_id`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8;


insert into `city` (`city_id`, `city_name`, `country_id`) values(1,'西安',1);
insert into `city` (`city_id`, `city_name`, `country_id`) values(2,'NewYork',2);
insert into `city` (`city_id`, `city_name`, `country_id`) values(3,'北京',1);
insert into `city` (`city_id`, `city_name`, `country_id`) values(4,'上海',1);

insert into `country` (`country_id`, `country_name`) values(1,'China');
insert into `country` (`country_id`, `country_name`) values(2,'America');
insert into `country` (`country_id`, `country_name`) values(3,'Japan');
insert into `country` (`country_id`, `country_name`) values(4,'UK');
2.5.1 創建索引

語法 :

CREATE 	[UNIQUE|FULLTEXT|SPATIAL]  INDEX index_name 
[USING  index_type]
ON tbl_name(index_col_name,...)


index_col_name : column_name[(length)][ASC | DESC]

示例 : 為city表中的city_name欄位創建索引 ;


?

?

2.5.2 查看索引

語法:

show index  from  table_name;

示例:查看city表中的索引資訊;

2.5.3 洗掉索引

語法 :

DROP  INDEX  index_name  ON  tbl_name;

示例 : 想要洗掉city表上的索引idx_city_name,可以操作如下:

2.5.4 ALTER命令
1). alter  table  tb_name  add  primary  key(column_list); 

	該陳述句添加一個主鍵,這意味著索引值必須是唯一的,且不能為NULL
	
2). alter  table  tb_name  add  unique index_name(column_list);
	
	這條陳述句創建索引的值必須是唯一的(除了NULL外,NULL可能會出現多次)
	
3). alter  table  tb_name  add  index index_name(column_list);

	添加普通索引, 索引值可以出現多次,
	
4). alter  table  tb_name  add  fulltext  index_name(column_list);
	
	該陳述句指定了索引為FULLTEXT, 用于全文索引
	

2.6 索引設計原則

? 索引的設計可以遵循一些已有的原則,創建索引的時候請盡量考慮符合這些原則,便于提升索引的使用效率,更高效的使用索引,

  • 對查詢頻次較高,且資料量比較大的表建立索引,

  • 索引欄位的選擇,最佳候選列應當從where子句的條件中提取,如果where子句中的組合比較多,那么應當挑選最常用、過濾效果最好的列的組合,

  • 使用唯一索引,區分度越高,使用索引的效率越高,

  • 索引可以有效的提升查詢資料的效率,但索引數量不是多多益善,索引越多,維護索引的代價自然也就水漲船高,對于插入、更新、洗掉等DML操作比較頻繁的表來說,索引過多,會引入相當高的維護代價,降低DML操作的效率,增加相應操作的時間消耗,另外索引過多的話,MySQL也會犯選擇困難病,雖然最終仍然會找到一個可用的索引,但無疑提高了選擇的代價,

  • 使用短索引,索引創建之后也是使用硬碟來存盤的,因此提升索引訪問的I/O效率,也可以提升總體的訪問效率,假如構成索引的欄位總長度比較短,那么在給定大小的存盤塊內可以存盤更多的索引值,相應的可以有效的提升MySQL訪問索引的I/O效率,

  • 利用最左前綴,N個列組合而成的組合索引,那么相當于是創建了N個索引,如果查詢時where子句中使用了組成該索引的前幾個欄位,那么這條查詢SQL可以利用組合索引來提升查詢效率,

    創建復合索引:
    
    	CREATE INDEX idx_name_email_status ON tb_seller(NAME,email,STATUS);
    
    就相當于
    	對name 創建索引 ;
    	對name , email 創建了索引 ;
    	對name , email, status 創建了索引 ;
    

3. 視圖

3.1 視圖概述

? 視圖(View)是一種虛擬存在的表,視圖并不在資料庫中實際存在,行和列資料來自定義視圖的查詢中使用的表,并且是在使用視圖時動態生成的,通俗的講,視圖就是一條SELECT陳述句執行后回傳的結果集,所以我們在創建視圖的時候,主要的作業就落在創建這條SQL查詢陳述句上,

視圖相對于普通的表的優勢主要包括以下幾項,

  • 簡單:使用視圖的用戶完全不需要關心后面對應的表的結構、關聯條件和篩選條件,對用戶來說已經是過濾好的復合條件的結果集,
  • 安全:使用視圖的用戶只能訪問他們被允許查詢的結果集,對表的權限管理并不能限制到某個行某個列,但是通過視圖就可以簡單的實作,
  • 資料獨立:一旦視圖的結構確定了,可以屏蔽表結構變化對用戶的影響,源表增加列對視圖沒有影響;源表修改列名,則可以通過修改視圖來解決,不會造成對訪問者的影響,

3.2 創建或者修改視圖

創建視圖的語法為:

CREATE [OR REPLACE] [ALGORITHM = {UNDEFINED | MERGE | TEMPTABLE}]

VIEW view_name [(column_list)]

AS select_statement

[WITH [CASCADED | LOCAL] CHECK OPTION]

修改視圖的語法為:

ALTER [ALGORITHM = {UNDEFINED | MERGE | TEMPTABLE}]

VIEW view_name [(column_list)]

AS select_statement

[WITH [CASCADED | LOCAL] CHECK OPTION]
選項 : 
	WITH [CASCADED | LOCAL] CHECK OPTION 決定了是否允許更新資料使記錄不再滿足視圖的條件,
	
	LOCAL : 只要滿足本視圖的條件就可以更新,
	CASCADED : 必須滿足所有針對該視圖的所有視圖的條件才可以更新, 默認值.

示例 , 創建city_country_view視圖 , 執行如下SQL :

create or replace view city_country_view 
as 
select t.*,c.country_name from country c , city t where c.country_id = t.country_id;

查詢視圖 :

3.3 查看視圖

? 從 MySQL 5.1 版本開始,使用 SHOW TABLES 命令的時候不僅顯示表的名字,同時也會顯示視圖的名字,而不存在單獨顯示視圖的 SHOW VIEWS 命令,

同樣,在使用 SHOW TABLE STATUS 命令的時候,不但可以顯示表的資訊,同時也可以顯示視圖的資訊,

如果需要查詢某個視圖的定義,可以使用 SHOW CREATE VIEW 命令進行查看 :

3.4 洗掉視圖

語法 :

DROP VIEW [IF EXISTS] view_name [, view_name] ...[RESTRICT | CASCADE]	

示例 , 洗掉視圖city_country_view :

DROP VIEW city_country_view ;

4. 存盤程序和函式

4.1 存盤程序和函式概述

? 存盤程序和函式是 事先經過編譯并存盤在資料庫中的一段 SQL 陳述句的集合,呼叫存盤程序和函式可以簡化應用開發人員的很多作業,減少資料在資料庫和應用服務器之間的傳輸,對于提高資料處理的效率是有好處的,

? 存盤程序和函式的區別在于函式必須有回傳值,而存盤程序沒有,

? 函式 : 是一個有回傳值的程序 ;

? 程序 : 是一個沒有回傳值的函式 ;

4.2 創建存盤程序

CREATE PROCEDURE procedure_name ([proc_parameter[,...]])
begin
	-- SQL陳述句
end ;

示例 :

delimiter $

create procedure pro_test1()
begin
	select 'Hello Mysql' ;
end$

delimiter ;

知識小貼士

DELIMITER

? 該關鍵字用來宣告SQL陳述句的分隔符 , 告訴 MySQL 解釋器,該段命令是否已經結束了,mysql是否可以執行了,默認情況下,delimiter是分號;,在命令列客戶端中,如果有一行命令以分號結束,那么回車后,mysql將會執行該命令,

4.3 呼叫存盤程序

call procedure_name() ;	

4.4 查看存盤程序

-- 查詢db_name資料庫中的所有的存盤程序
select name from mysql.proc where db='db_name';


-- 查詢存盤程序的狀態資訊
show procedure status;


-- 查詢某個存盤程序的定義
show create procedure test.pro_test1 \G;

4.5 洗掉存盤程序

DROP PROCEDURE  [IF EXISTS] sp_name ;

4.6 語法

存盤程序是可以編程的,意味著可以使用變數,運算式,控制結構 , 來完成比較復雜的功能,

4.6.1 變數
  • DECLARE

通過 DECLARE 可以定義一個區域變數,該變數的作用范圍只能在 BEGIN…END 塊中,

DECLARE var_name[,...] type [DEFAULT value]

示例 :

 delimiter $

 create procedure pro_test2() 
 begin 
 	declare num int default 5;
 	select num+ 10; 
 end$

 delimiter ; 
  • SET

直接賦值使用 SET,可以賦常量或者賦運算式,具體語法如下:

  SET var_name = expr [, var_name = expr] ...

示例 :

  DELIMITER $
  
  CREATE  PROCEDURE pro_test3()
  BEGIN
  	DECLARE NAME VARCHAR(20);
  	SET NAME = 'MYSQL';
  	SELECT NAME ;
  END$
  
  DELIMITER ;

也可以通過select ... into 方式進行賦值操作 :

DELIMITER $

CREATE  PROCEDURE pro_test5()
BEGIN
	declare  countnum int;
	select count(*) into countnum from city;
	select countnum;
END$

DELIMITER ;
4.6.2 if條件判斷

語法結構 :

if search_condition then statement_list

	[elseif search_condition then statement_list] ...
	
	[else statement_list]
	
end if;

需求:

根據定義的身高變數,判定當前身高的所屬的身材型別 

	180 及以上 ----------> 身材高挑

	170 - 180  ---------> 標準身材

	170 以下  ----------> 一般身材

示例 :

delimiter $

create procedure pro_test6()
begin
  declare  height  int  default  175; 
  declare  description  varchar(50);
  
  if  height >= 180  then
    set description = '身材高挑';
  elseif height >= 170 and height < 180  then
    set description = '標準身材';
  else
    set description = '一般身材';
  end if;
  
  select description ;
end$

delimiter ;

呼叫結果為 :

4.6.3 傳遞引數

語法格式 :

create procedure procedure_name([in/out/inout] 引數名   引數型別)
...


IN :   該引數可以作為輸入,也就是需要呼叫方傳入值 , 默認
OUT:   該引數作為輸出,也就是該引數可以作為回傳值
INOUT: 既可以作為輸入引數,也可以作為輸出引數

IN - 輸入

需求 :

根據定義的身高變數,判定當前身高的所屬的身材型別 

示例 :

delimiter $

create procedure pro_test5(in height int)
begin
    declare description varchar(50) default '';
  if height >= 180 then
    set description='身材高挑';
  elseif height >= 170 and height < 180 then
    set description='標準身材';
  else
    set description='一般身材';
  end if;
  select concat('身高 ', height , '對應的身材型別為:',description);
end$

delimiter ;

OUT-輸出

需求 :

根據傳入的身高變數,獲取當前身高的所屬的身材型別  

示例:

create procedure pro_test5(in height int , out description varchar(100))
begin
  if height >= 180 then
    set description='身材高挑';
  elseif height >= 170 and height < 180 then
    set description='標準身材';
  else
    set description='一般身材';
  end if;
end$	

呼叫:

call pro_test5(168, @description)$

select @description$

小知識

@description : 這種變數要在變數名稱前面加上“@”符號,叫做用戶會話變數,代表整個會話程序他都是有作用的,這個類似于全域變數一樣,

@@global.sort_buffer_size : 這種在變數前加上 "@@" 符號, 叫做 系統變數

4.6.4 case結構

語法結構 :

方式一 : 

CASE case_value

  WHEN when_value THEN statement_list
  
  [WHEN when_value THEN statement_list] ...
  
  [ELSE statement_list]
  
END CASE;


方式二 : 

CASE

  WHEN search_condition THEN statement_list
  
  [WHEN search_condition THEN statement_list] ...
  
  [ELSE statement_list]
  
END CASE;

需求:

給定一個月份, 然后計算出所在的季度

示例 :

delimiter $


create procedure pro_test9(month int)
begin
  declare result varchar(20);
  case 
    when month >= 1 and month <=3 then 
      set result = '第一季度';
    when month >= 4 and month <=6 then 
      set result = '第二季度';
    when month >= 7 and month <=9 then 
      set result = '第三季度';
    when month >= 10 and month <=12 then 
      set result = '第四季度';
  end case;
  
  select concat('您輸入的月份為 :', month , ' , 該月份為 : ' , result) as content ;
  
end$


delimiter ;
4.6.5 while回圈

語法結構:

while search_condition do

	statement_list
	
end while;

需求:

計算從1加到n的值

示例 :

delimiter $

create procedure pro_test8(n int)
begin
  declare total int default 0;
  declare num int default 1;
  while num<=n do
    set total = total + num;
	set num = num + 1;
  end while;
  select total;
end$

delimiter ;
4.6.6 repeat結構

有條件的回圈控制陳述句, 當滿足條件的時候退出回圈 ,while 是滿足條件才執行,repeat 是滿足條件就退出回圈,

語法結構 :

REPEAT

  statement_list

  UNTIL search_condition

END REPEAT;

需求:

計算從1加到n的值

示例 :

delimiter $

create procedure pro_test10(n int)
begin
  declare total int default 0;
  
  repeat 
    set total = total + n;
    set n = n - 1;
    until n=0  
  end repeat;
  
  select total ;
  
end$


delimiter ;
4.6.7 loop陳述句

LOOP 實作簡單的回圈,退出回圈的條件需要使用其他的陳述句定義,通常可以使用 LEAVE 陳述句實作,具體語法如下:

[begin_label:] LOOP

  statement_list

END LOOP [end_label]

如果不在 statement_list 中增加退出回圈的陳述句,那么 LOOP 陳述句可以用來實作簡單的死回圈,

4.6.8 leave陳述句

用來從標注的流程構造中退出,通常和 BEGIN ... END 或者回圈一起使用,下面是一個使用 LOOP 和 LEAVE 的簡單例子 , 退出回圈:

delimiter $

CREATE PROCEDURE pro_test11(n int)
BEGIN
  declare total int default 0;
  
  ins: LOOP
    
    IF n <= 0 then
      leave ins;
    END IF;
    
    set total = total + n;
    set n = n - 1;
  	
  END LOOP ins;
  
  select total;
END$

delimiter ;
4.6.9 游標/游標

游標是用來存盤查詢結果集的資料型別 , 在存盤程序和函式中可以使用游標對結果集進行回圈的處理,游標的使用包括游標的宣告、OPEN、FETCH 和 CLOSE,其語法分別如下,

宣告游標:

DECLARE cursor_name CURSOR FOR select_statement ;

OPEN 游標:

OPEN cursor_name ;

FETCH 游標:

FETCH cursor_name INTO var_name [, var_name] ...

CLOSE 游標:

CLOSE cursor_name ;

示例 :

初始化腳本:

create table emp(
  id int(11) not null auto_increment ,
  name varchar(50) not null comment '姓名',
  age int(11) comment '年齡',
  salary int(11) comment '薪水',
  primary key(`id`)
)engine=innodb default charset=utf8 ;

insert into emp(id,name,age,salary) values(null,'金毛獅王',55,3800),(null,'白眉鷹王',60,4000),(null,'青翼蝠王',38,2800),(null,'紫衫龍王',42,1800);

-- 查詢emp表中資料, 并逐行獲取進行展示
create procedure pro_test11()
begin
  declare e_id int(11);
  declare e_name varchar(50);
  declare e_age int(11);
  declare e_salary int(11);
  declare emp_result cursor for select * from emp;
  
  open emp_result;
  
  fetch emp_result into e_id,e_name,e_age,e_salary;
  select concat('id=',e_id , ', name=',e_name, ', age=', e_age, ', 薪資為: ',e_salary);
  
  fetch emp_result into e_id,e_name,e_age,e_salary;
  select concat('id=',e_id , ', name=',e_name, ', age=', e_age, ', 薪資為: ',e_salary);
  
  fetch emp_result into e_id,e_name,e_age,e_salary;
  select concat('id=',e_id , ', name=',e_name, ', age=', e_age, ', 薪資為: ',e_salary);
  
  fetch emp_result into e_id,e_name,e_age,e_salary;
  select concat('id=',e_id , ', name=',e_name, ', age=', e_age, ', 薪資為: ',e_salary);
  
  fetch emp_result into e_id,e_name,e_age,e_salary;
  select concat('id=',e_id , ', name=',e_name, ', age=', e_age, ', 薪資為: ',e_salary);
  
  close emp_result;
end$

通過回圈結構 , 獲取游標中的資料 :

DELIMITER $

create procedure pro_test12()
begin
  DECLARE id int(11);
  DECLARE name varchar(50);
  DECLARE age int(11);
  DECLARE salary int(11);
  DECLARE has_data int default 1;
  
  DECLARE emp_result CURSOR FOR select * from emp;
  DECLARE EXIT HANDLER FOR NOT FOUND set has_data = https://www.cnblogs.com/lovelywcc/p/0;
  
  open emp_result;
  
  repeat
    fetch emp_result into id , name , age , salary;
    select concat('id為',id, ', name 為' ,name , ', age為 ' ,age , ', 薪水為: ', salary);
    until has_data = https://www.cnblogs.com/lovelywcc/p/0
  end repeat;
  
  close emp_result;
end$

DELIMITER ; 

4.7 存盤函式

語法結構:

CREATE FUNCTION function_name([param type ... ]) 
RETURNS type 
BEGIN
	...
END;

案例 :

定義一個存盤程序, 請求滿足條件的總記錄數 ;


delimiter $

create function count_city(countryId int)
returns int
begin
  declare cnum int ;
  
  select count(*) into cnum from city where country_id = countryId;
  
  return cnum;
end$

delimiter ;

呼叫:

select count_city(1);

select count_city(2);

5. 觸發器

5.1 介紹

觸發器是與表有關的資料庫物件,指在 insert/update/delete 之前或之后,觸發并執行觸發器中定義的SQL陳述句集合,觸發器的這種特性可以協助應用在資料庫端確保資料的完整性 , 日志記錄 , 資料校驗等操作 ,

使用別名 OLD 和 NEW 來參考觸發器中發生變化的記錄內容,這與其他的資料庫是相似的,現在觸發器還只支持行級觸發,不支持陳述句級觸發,

觸發器型別 NEW 和 OLD的使用
INSERT 型觸發器 NEW 表示將要或者已經新增的資料
UPDATE 型觸發器 OLD 表示修改之前的資料 , NEW 表示將要或已經修改后的資料
DELETE 型觸發器 OLD 表示將要或者已經洗掉的資料

5.2 創建觸發器

語法結構 :

create trigger trigger_name 

before/after insert/update/delete

on tbl_name 

[ for each row ]  -- 行級觸發器

begin

	trigger_stmt ;

end;

示例

需求

通過觸發器記錄 emp 表的資料變更日志 , 包含增加, 修改 , 洗掉 ;

首先創建一張日志表 :

create table emp_logs(
  id int(11) not null auto_increment,
  operation varchar(20) not null comment '操作型別, insert/update/delete',
  operate_time datetime not null comment '操作時間',
  operate_id int(11) not null comment '操作表的ID',
  operate_params varchar(500) comment '操作引數',
  primary key(`id`)
)engine=innodb default charset=utf8;

創建 insert 型觸發器,完成插入資料時的日志記錄 :

DELIMITER $

create trigger emp_logs_insert_trigger
after insert 
on emp 
for each row 
begin
  insert into emp_logs (id,operation,operate_time,operate_id,operate_params) values(null,'insert',now(),new.id,concat('插入后(id:',new.id,', name:',new.name,', age:',new.age,', salary:',new.salary,')'));	
end $

DELIMITER ;

創建 update 型觸發器,完成更新資料時的日志記錄 :

DELIMITER $

create trigger emp_logs_update_trigger
after update 
on emp 
for each row 
begin
  insert into emp_logs (id,operation,operate_time,operate_id,operate_params) values(null,'update',now(),new.id,concat('修改前(id:',old.id,', name:',old.name,', age:',old.age,', salary:',old.salary,') , 修改后(id',new.id, 'name:',new.name,', age:',new.age,', salary:',new.salary,')'));                                                                      
end $

DELIMITER ;

創建delete 行的觸發器 , 完成洗掉資料時的日志記錄 :

DELIMITER $

create trigger emp_logs_delete_trigger
after delete 
on emp 
for each row 
begin
  insert into emp_logs (id,operation,operate_time,operate_id,operate_params) values(null,'delete',now(),old.id,concat('洗掉前(id:',old.id,', name:',old.name,', age:',old.age,', salary:',old.salary,')'));                                                                      
end $

DELIMITER ;

測驗:

insert into emp(id,name,age,salary) values(null, '光明左使',30,3500);
insert into emp(id,name,age,salary) values(null, '光明右使',33,3200);

update emp set age = 39 where id = 3;

delete from emp where id = 5;

5.3 洗掉觸發器

語法結構 :

drop trigger [schema_name.]trigger_name

如果沒有指定 schema_name,默認為當前資料庫 ,

5.4 查看觸發器

可以通過執行 SHOW TRIGGERS 命令查看觸發器的狀態、語法等資訊,

語法結構 :

show triggers ;

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

標籤:Java

上一篇:Java連載138-資料庫洗掉資料以及編譯預處理

下一篇:Spring Batch遠程磁區的本地Jar包模式

標籤雲
其他(157675) Python(38076) JavaScript(25376) Java(17977) C(15215) 區塊鏈(8255) C#(7972) AI(7469) 爪哇(7425) MySQL(7132) html(6777) 基礎類(6313) sql(6102) 熊猫(6058) PHP(5869) 数组(5741) R(5409) Linux(5327) 反应(5209) 腳本語言(PerlPython)(5129) 非技術區(4971) Android(4554) 数据框(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技术(1958) Web開發(1951) python-3.x(1918) HtmlCss(1915) 弹簧靴(1913) C++(1909) xml(1889) PostgreSQL(1872) .NETCore(1853) 谷歌表格(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
最新发布
  • Rust中的智能指標:Box<T> Rc<T> Arc<T> Cell<T> RefCell<T> Weak

    Rust中的智能指標是什么 智能指標(smart pointers)是一類資料結構,是擁有資料所有權和額外功能的指標。是指標的進一步發展 指標(pointer)是一個包含記憶體地址的變數的通用概念。這個地址參考,或 ” 指向”(points at)一些其 他資料 。參考以 & 符號為標志并借用了他們所 ......

    uj5u.com 2023-04-20 07:24:10 more
  • Java的值傳遞和參考傳遞

    值傳遞不會改變本身,參考傳遞(如果傳遞的值需要實體化到堆里)如果發生修改了會改變本身。 1.基本資料型別都是值傳遞 package com.example.basic; public class Test { public static void main(String[] args) { int ......

    uj5u.com 2023-04-20 07:24:04 more
  • [2]SpinalHDL教程——Scala簡單入門

    第一個 Scala 程式 shell里面輸入 $ scala scala> 1 + 1 res0: Int = 2 scala> println("Hello World!") Hello World! 檔案形式 object HelloWorld { /* 這是我的第一個 Scala 程式 * 以 ......

    uj5u.com 2023-04-20 07:23:58 more
  • 理解函式指標和回呼函式

    理解 函式指標 指向函式的指標。比如: 理解函式指標的偽代碼 void (*p)(int type, char *data); // 定義一個函式指標p void func(int type, char *data); // 宣告一個函式func p = func; // 將指標p指向函式func ......

    uj5u.com 2023-04-20 07:23:52 more
  • Django筆記二十五之資料庫函式之日期函式

    本文首發于公眾號:Hunter后端 原文鏈接:Django筆記二十五之資料庫函式之日期函式 日期函式主要介紹兩個大類,Extract() 和 Trunc() Extract() 函式作用是提取日期,比如我們可以提取一個日期欄位的年份,月份,日等資料 Trunc() 的作用則是截取,比如 2022-0 ......

    uj5u.com 2023-04-20 07:23:45 more
  • 一天吃透JVM面試八股文

    什么是JVM? JVM,全稱Java Virtual Machine(Java虛擬機),是通過在實際的計算機上仿真模擬各種計算機功能來實作的。由一套位元組碼指令集、一組暫存器、一個堆疊、一個垃圾回收堆和一個存盤方法域等組成。JVM屏蔽了與作業系統平臺相關的資訊,使得Java程式只需要生成在Java虛擬機 ......

    uj5u.com 2023-04-20 07:23:31 more
  • 使用Java接入小程式訂閱訊息!

    更新完微信服務號的模板訊息之后,我又趕緊把微信小程式的訂閱訊息給實作了!之前我一直以為微信小程式也是要企業才能申請,沒想到小程式個人就能申請。 訊息推送平臺🔥推送下發【郵件】【短信】【微信服務號】【微信小程式】【企業微信】【釘釘】等訊息型別。 https://gitee.com/zhongfuch ......

    uj5u.com 2023-04-20 07:22:59 more
  • java -- 緩沖流、轉換流、序列化流

    緩沖流 緩沖流, 也叫高效流, 按照資料型別分類: 位元組緩沖流:BufferedInputStream,BufferedOutputStream 字符緩沖流:BufferedReader,BufferedWriter 緩沖流的基本原理,是在創建流物件時,會創建一個內置的默認大小的緩沖區陣列,通過緩沖 ......

    uj5u.com 2023-04-20 07:22:49 more
  • Java-SpringBoot-Range請求頭設定實作視頻分段傳輸

    老實說,人太懶了,現在基本都不喜歡寫筆記了,但是網上有關Range請求頭的文章都太水了 下面是抄的一段StackOverflow的代碼...自己大修改過的,寫的注釋挺全的,應該直接看得懂,就不解釋了 寫的不好...只是希望能給視頻網站開發的新手一點點幫助吧. 業務場景:視頻分段傳輸、視頻多段傳輸(理 ......

    uj5u.com 2023-04-20 07:22:42 more
  • Windows 10開發教程_編程入門自學教程_菜鳥教程-免費教程分享

    教程簡介 Windows 10開發入門教程 - 從簡單的步驟了解Windows 10開發,從基本到高級概念,包括簡介,UWP,第一個應用程式,商店,XAML控制元件,資料系結,XAML性能,自適應設計,自適應UI,自適應代碼,檔案管理,SQLite資料庫,應用程式到應用程式通信,應用程式本地化,應用程式 ......

    uj5u.com 2023-04-20 07:22:35 more