MySQL中的視圖
一、什麼是視圖
通俗的講,視圖就是一條SELECT語句執行後返回的結果集。所以我們在建立視圖的時候,主要的工作就落在建立這條SQL查詢語句上。
二、視圖的特性
視圖是對若干張基本表的引用,一張虛表,查詢語句執行的結果,不儲存具體的資料(基本表資料發生了改變,視圖也會跟著改變);
可以跟基本表一樣,進行增刪改查操作(ps:增刪改操作有條件限制);
三、視圖的作用
方便操作,特別是查詢操作,減少複雜的SQL語句,增強可讀性;
更加安全,資料庫授權命令不能限定到特定行和特定列,但是通過合理建立視圖,可以把許可權限定到行列層級;
四、使用場合
許可權控制的時候,不希望使用者訪問表中某些含敏感資訊的列,比如salary...
關鍵資訊來源於多個複雜關聯表,可以建立視圖提取我們需要的資訊,簡化操作;
五、視圖執行個體1-建立視圖及查詢資料操作
現有三張表:使用者(user)、課程(course)、使用者課程中間表(user_course),表結構及資料如下:
表定義:
-- ----------------------------
-- Table structure for `course`
-- ----------------------------
DROP TABLE IF EXISTS `course`;
CREATE TABLE `course` (
`id` bigint(20) NOT NULL AUTO_INCREMENT,
`name` varchar(200) NOT NULL,
`description` varchar(500) NOT NULL,
PRIMARY KEY (`id`)
) ENGINE=InnoDB AUTO_INCREMENT=4 DEFAULT CHARSET=utf8;
-- ----------------------------
-- Records of course
-- ----------------------------
INSERT INTO `course` VALUES ('1', 'JAVA', 'JAVA課程');
INSERT INTO `course` VALUES ('2', 'C++', 'C++課程');
INSERT INTO `course` VALUES ('3', 'C語言', 'C語言課程');
-- ----------------------------
-- Table structure for `user`
-- ----------------------------
DROP TABLE IF EXISTS `user`;
CREATE TABLE `user` (
`id` bigint(20) NOT NULL AUTO_INCREMENT,
`account` varchar(255) NOT NULL,
`name` varchar(255) NOT NULL,
`address` varchar(255) DEFAULT NULL,
`others` varchar(200) DEFAULT NULL,
`others2` varchar(200) DEFAULT NULL,
PRIMARY KEY (`id`)
) ENGINE=InnoDB AUTO_INCREMENT=4 DEFAULT CHARSET=utf8;
-- ----------------------------
-- Records of user
-- ----------------------------
INSERT INTO `user` VALUES ('1', 'user1', '小陳', '美國', '1', '1');
INSERT INTO `user` VALUES ('2', 'user2', '小張', '日本', '2', '2');
INSERT INTO `user` VALUES ('3', 'user3', '小王', '中國', '3', '3');
-- ----------------------------
-- Table structure for `user_course`
-- ----------------------------
DROP TABLE IF EXISTS `user_course`;
CREATE TABLE `user_course` (
`id` bigint(20) NOT NULL AUTO_INCREMENT,
`userid` bigint(20) NOT NULL,
`courseid` bigint(20) NOT NULL,
PRIMARY KEY (`id`)
) ENGINE=InnoDB AUTO_INCREMENT=7 DEFAULT CHARSET=utf8;
-- ----------------------------
-- Records of user_course
-- ----------------------------
INSERT INTO `user_course` VALUES ('1', '1', '2');
INSERT INTO `user_course` VALUES ('2', '1', '3');
INSERT INTO `user_course` VALUES ('3', '2', '1');
INSERT INTO `user_course` VALUES ('4', '2', '2');
INSERT INTO `user_course` VALUES ('5', '2', '3');
INSERT INTO `user_course` VALUES ('6', '3', '2');
表資料:
這時,當我們想要查詢小張上的所以課程相關資訊的時候,需要這樣寫一條長長的SQL語句,如下:
SELECT
`uc`.`id` AS `id`,
`u`.`name` AS `username`,
`c`.`name` AS `coursename`
FROM
`user` `u`
LEFT JOIN `user_course` `uc` ON ((`u`.`id` = `uc`.`userid`))
LEFT JOIN `course` `c` ON ((`uc`.`courseid` = `c`.`id`))
WHERE
u.`name` = '小張'
但是我們可以通過視圖簡化操作,例如我們建立視圖view_user_course如下:
-- ----------------------------
-- View structure for `view_user_course`
-- ----------------------------
DROP VIEW
IF EXISTS `view_user_course`;
CREATE ALGORITHM = UNDEFINED
DEFINER = `root`@`localhost`
SQL SECURITY DEFINER
VIEW `view_user_course` AS (
SELECT
`uc`.`id` AS `id`,
`u`.`name` AS `username`,
`c`.`name` AS `coursename`
FROM
(
(
`user` `u`
LEFT JOIN `user_course` `uc` ON ((`u`.`id` = `uc`.`userid`))
)
LEFT JOIN `course` `c` ON ((`uc`.`courseid` = `c`.`id`))
)
);
幾點說明(MySQL中的視圖在標準SQL的基礎之上做了擴充):
ALGORITHM=UNDEFINED:指定視圖的處理演算法;
DEFINER=`root`@`localhost`:指定視圖建立者;
SQL SECURITY DEFINER:指定視圖查詢資料時的安全驗證方式;
建立好視圖之後,我們可以直接用以下SQL語句在視圖上查詢小張上的所以課程相關資訊,同樣可以得到所需結果:
SELECT
vuc.username,
vuc.coursename
FROM
view_user_course vuc
WHERE
vuc.username = '小張'
六、視圖執行個體2-增刪改資料操作
繼續,我們可以嘗試在視圖view_user_course上做增刪改資料操作,如下:
update view_user_course set username='test',coursename='JAVASCRIPT' where id=3
遺憾的是操作失敗,提示錯誤資訊如下:
[SQL] update view_user_course set username='test',coursename='JAVASCRIPT' where id=3
[Err] 1393 - Can not modify more than one base table through a join view 'demo.view_user_course'
因為不能在一張由多張關聯表串連而成的視圖上做同時修改兩張表的操作;
那麼哪些操作可以在視圖上進行呢?
視圖與表是一對一關聯性情況:如果沒有其它約束(如視圖中沒有的欄位,在基本表中是必要欄位情況),是可以進行增刪改資料操作;
如我們建立使用者關鍵資訊視圖view_user_keyinfo,如下:
-- ----------------------------
-- View structure for `view_user_keyinfo`
-- ----------------------------
DROP VIEW
IF EXISTS `view_user_keyinfo`;
CREATE ALGORITHM = UNDEFINED DEFINER = `root`@`localhost` SQL SECURITY DEFINER VIEW `view_user_keyinfo` AS SELECT
`u`.`id` AS `id`,
`u`.`account` AS `account`,
`u`.`name` AS `username`
FROM
`user` `u`;
進行增刪改操作如下,操作成功(注意user表中的其它欄位要允許為空白,否則操作失敗):
INSERT INTO view_user_keyinfo (account, username)
VALUES
('test1', 'test1');
DELETE
FROM
view_user_keyinfo
WHERE
username = 'test1';
UPDATE view_user_keyinfo
SET username = 'updateuser'
WHERE
id = 1
視圖與表是一對多關聯性情況:如果只修改一張表的資料,且沒有其它約束(如視圖中沒有的欄位,在基本表中是必要欄位情況),是可以進行改資料操作,如以下語句,操作成功;
update view_user_course set coursename='JAVA' where id=1;
update view_user_course set username='test2' where id=3;
以下操作失敗:
delete from view_user_course where id=3;
insert into view_user_course(username, coursename) VALUES('2','3');
七、其它
視圖中的查詢語句效能要調到最優;
修改操作時要小心,不經意間你已經修改了基本表裡的多條資料;
其它效能相關方面待實踐體會...
本文永久更新連結地址: