mysql update语句多表关联,update后可接两张表吗,
mysql update语句多表关联,update后可接两张表吗,详细介绍
本文目录一览: mysql如何实现多表联合更新
案例如下:现在有一张user表,还有一张userinfo表。想将userinfo表里面的一个remark字段更新到user表中的remark字段。那么怎么操作呢?这里就要用到MySQL的多表联合更新。update user,userinfo set user.remark=userinfo.remark where user.id=userinfo.id注意,这里user表和userinfo表,我都假设有一个id是唯一键来进行关联的。
sql mysql多表如何关联查询
要实现关联查询,这些表之间必须有关系:
举例说明:
如果三个表tableA、tableB、tableC满足关联条件
tableA.id = tableB.pid
tableB.cid = tableC.did
就可以通过LEFT JOIN关联查询,查出你要查找的关联信息
$sql = 'Select A.*, B.*, C.* From tableA as A LEFT JOIN tableB on A.id = B.pid LEFT JOIN tableC on B.cid = C.did Where A.id =8';
一使用SELECT子句进行多表查询
SELECT 字段名 FROM 表1,表2 … WHERE 表1.字段 = 表2.字段 AND 其它查询条件
SELECT a.id,a.name,a.address,a.date,b.math,b.english,b.chinese FROM tb_demo065_tel AS b,tb_demo065 AS a WHERE a.id=b.id
注:在上面的的代码中,以两张表的id字段信息相同作为条件建立两表关联,但在实际开发中不应该这样使用,最好用主外键约束来实现
一使用SELECT子句进行多表查询
SELECT 字段名 FROM 表1,表2 … WHERE 表1.字段 = 表2.字段 AND 其它查询条件
SELECT a.id,a.name,a.address,a.date,b.math,b.english,b.chinese FROM tb_demo065_tel AS b,tb_demo065 AS a WHERE a.id=b.id
注:在上面的的代码中,以两张表的id字段信息相同作为条件建立两表关联,但在实际开发中不应该这样使用,最好用主外键约束来实现
二使用表的别名进行多表查询
如:SELECT a.id,a.name,a.address,b.math,b.english,b.chinese FROM tb_demo065 a,tb_demo065_tel b WHERE a.id=b.id AND b.id='$_POST[textid]'
SQL语言中,可以通过两种方式为表指定别名
第一种是通过关键字AS指定,如
SELECT a.id,a.name,a.address,b.math,b.english,b.chinese FROM tb_demo065 AS a,tb_demo065_tel AS b WHERE a.id=b.id
第二种是在表名后直接加表的别名实现
SELECT a.id,a.name,a.address,b.math,b.english,b.chinese FROM tb_demo065 a,tb_demo065_tel b WHERE a.id=b.id
使用表的别名应注意几下几点
(1)别名通常是一个缩短了的表名,用于在连接中引用表中的特定列,如果连接中的多个表中有相同的名称列存在,必须用表名或表的别名限定列名
(2)如果定义了表的别名就不能再使用表名
三合并多个结果集
SQL语言中,可以通过UNION 或 ALL将多个SELECT语句的查询结果合并输出,这两个关键字的使用说明如下:
UNION:利用该关键字可以将多个SELECT 语句的查询结果合并输出,并删除重复行
ALL:利用该关键字可以将多个SELECT 语句的查询结果合并输出,但不会删除重复行
在使用UNION或ALL关键字将多个表合并输出时,查询结果必须具有相同的结构并且数据类型必须兼容,另外使用UNION时两张表的字段数量也必须相同,否则会提示SQL语句有错误。
e.x:SELECT id,name,pwd FROM tb_demo067 UNION SELECT uid,price,date FROM tb_demo067_tel
四简单嵌套查询
子查询:子查询是一个SELECT查询,返回单个值且嵌套在SELECT、INSERT、UPDATE和DELETE语句或其它查询语句中,任何可以使用表达式的地方都可以使用子查询.
SELECT id,name,sex,date FROM tb_demo068 WHERE id in(SELECT id FROM tb_demo068 WHERE id='$_POST[test]')
内连接:把查询结果作为WHERE子句的查询条件即称为内连接
五复杂的嵌套查询
多表之间的嵌套查询可以通过谓词IN实现,语法格式如下:
test_expression[NOT] IN{
subquery
}
参数说明:test_expression指SQL表达式,subquery包含某结果集的子查询
多表嵌套查询的原理:无论是多少张表进行嵌套,表与表之间一定存在某种关联,通过WHERE子句建立此种关联实现查询
六嵌套查询在查询统计中的应用
实现多表查询时,可以同时使用谓词ANY、SOME、ALL,这些谓词被称为定量比较谓词,可以和比较运算符联合使用,判断是否全部返回值都满足搜索条件.SOME和ANY谓词是存在量的,只注重是否有返回值满足搜索条件,这两个谓词的含义相同,可以替换使用;ALL谓词称为通用谓词,它只关心是否有谓词满足搜索要求.
SELECT * FROM tb_demo069_people WHERE uid IN(SELECT deptID FROM tb_demo069_dept WHERE deptName='$_POST[select]')
SELECT a.id,a.name FROM tb_demo067 AS a WHERE id<3)
>ANY 大于子查询中的某个值
>=ANY 大于等于子查询中的某个值
<=ANY 小于等于子查询中的某个值
=ANY 等于子查询中的某个值
!=ANY或<>ANY 不等于子查询中的某个值
>ALL 大于子查询中的所有值
>=ALL 大于等于子查询中的所有值
<=ALL 小于等于子查询中的所有值
=ALL 等于子查询中的所有值
!=ALL或<>ALL 不等于子查询中的所有值
七.使用子查询作派生的表
在实际项目开发过程中经常用到从一个信息较为完善的表中派生出一个只含有几个关键字段的信息表,通过子查询就可以来实现这一目标,如
SELECT people.name,people.chinese,people.math,people.english FROM (SELECT name,chinese,math,english FROM tb_demo071) AS people
注:子查询应遵循以下规则:
(1)由比较运算符引入的内层子查询只包含一个表达式或列名,在外层语句中的WHERE子句内命名的列必须与内层子查询命名的列兼容
(2)由不可更改的比较运算符引入的子查询(比较运算符后面不跟关键字ANY或ALL)不包括GROUP BY 或 HAVING子句,除非预先确定了成组或单个的值
(3)用EXISTS引入的SELECT列表一般都由*组成,不必指定列名
(4)子查询不能在内部处理其结果
八使用子查询作表达式
SELECT (SELECT AVG(chinese)FROM tb_demo071),(SELECT AVG(english)FROM tb_demo071),(SELECT AVG(math)FROM tb_demo071) FROM tb_demo071
注:在使用子查询时最好为列表项取个别名,这样可以方便用户在使用mysql_fetch_array()函数时为表项赋值,如
SELECT (SELECT AVG(chinese) FROM tb_demo071) AS yuwen ,(SELECT AVG(english) FROM tb_demo071) AS yingyu,(SELECT AVG(math) FROM tb_demo071) AS shuxue FROM tb_demo071
九使用子查询关联数据
SELECT * FROM tb_demo072_student WHERE id=(SELECT id FROM tb_demo072_class WHERE className = '$_POST[text]')
十多表联合查询
利用SQL语句中的UNION,可以将不同表中符合条件的数据信息显示在同一列中。
e.x:SELECT * FROM tb_demo074_student UNION SELECT * FROM tb_demo074_fasten
注:使用UNION时应注意以下两点:
(1)在使用UNION运算符组合的语句中,所有选择列表的表达式数目必须相同,如列名、算术表达式及聚合函数等
(2)在每个查询表中,对应列的数据结构必须一样。
十一对联合后的结果进行排序
为了UNION的运算兼容,要求所有SELECT语句都不能有ORDER BY语句,但有一种情况例外,那就是在最后一个SELECT语句中放置ORDER BY 子句实现结果的最终排序输出。
e.x:SELECT * FROM tb_demo074_student UNION SELECT * FROM tb_demo074_fasten ORDER BY id
使用UNION条件上相对比较苛刻,所以使用此语句时一定要注意两个表项数目和字段类型是否相同
十二条件联合语句
SELECT * FROM tb_demo076_BEIJING GROUP BY name HAVING name='人民邮电出版社' OR name='机械工业出版社' UNION SELECT * FROM tb_demo076_BEIJING GROUP BY name HAVING name <>'人民邮电出版社' AND name <>'机械工业再版社' ORDER BY id
上面语句应用了GROUP BY分组语句和HAVING语句实现条件联合查询。其实现目的是先保证将'人民邮电出版社'和'机械工业出版社'始终位于名单最前列,然后再输出其它的出版社
十三简单内连接查询
SELECT filedlist FROM table1 [INNER] JOIN table2 ON table1.column1 = table2.column1
其中,filedlist是要显示的字段,INNER表示表之间的连接方式为内连接,table1.column1=table2.column1用于指明两表间的连接条件,如:
SELECT a.name,a.address,a.date,b.chinese,b.math,b.english FROM tb_demo065 AS a INNER JOIN tb_demo065_tel AS b on a.id=b.id
十四复杂内连接查询
复杂的内连接查询是在基本的内连接查询的基础上再附加一些查询条件,如:
SELECT a.name,a.address,a.date,b.chinese,b.math,b.english FROM tb_demo065 AS a INNER JOIN tb_demo065_tel AS b on a.id=b.id WHERE b.id=(SELECT id FROM tb_demo065 WHERE tb_demo065.name='$_POST[text]')
总之,实现表与表之间的关联的本质是两表之间存在共同的数据项或者相同的数据项,通过WHERE 子句或内连接INNER JOIN … ON 语句将两表连接起来,实现查询
十五使用外连接实现多表联合查询
(1)LEFT OUTER JOIN表示表之间通过左连接方式相互连接,也可简写成LEFT JOIN,它是以左侧的表为基准故称左连接,左侧表中所有信息将被全部输出,而右侧表信息则只会输出符合条件的信息,对不符合条件的信息则返回NULL
e.x:SELECT a.name,a.address,b.math,b.english FROM tb_demo065 AS A LEFT OUTER JOIN tb_demo065_tel AS b ON a.id=b.id
(2)RIGHT OUTER JOIN表示表之间通过右连接方式相互连接,也可简写成RIGHT JOIN,它是以右侧的表为基准故称右连接,右侧表中所有信息将被全部输出,而左侧表信息则只会输出符合条件的信息,对不符合条件的信息则返回NULL
E.X:SELECT a.name,a.address,b.math,b.english FROM tb_demo065 AS A RIGHT OUTER JOIN tb_demo065_tel AS b ON a.id=b.id
十六利用IN或NOTIN关键字限定范围
e.x:SELECT * FROM tb_demo083 WHERE code IN(SELECT code FROM tb_demo083 WHERE code BETWEEN '$_POST[text1]' AND '$_POST[text2]')
利用IN可指定在范围内查询,若要求在某范围外查询可以用NOT IN代替它
十七由IN引入的关联子查询
e.x:SELECT * FROM tb_demo083 WHERE code IN(SELECT code FROM tb_demo083 WHERE code = '$_POST[text]')
十八利用HAVING语句过滤分组数据
HAVING子句用于指定组或聚合的搜索条件,HAVING通常与GROUP BY 语句一起使用,如果SQL语句中不含GROUP BY子句,则HAVING的行为与WHERE子句一样.
e.x:SELECT name,math FROM tb_demo083 GROUP BY id HAVING math > '95'
mysql怎么让2个表关联起来
1、创建测试表,
create table test_tbl1(id int, p_id int);
create table test_tbl2(id int, name varchar2(20));
2、插入测试数据
insert into test_tbl1 values(1001,1);
insert into test_tbl1 values(1002,1);
insert into test_tbl1 values(2001,2);
insert into test_tbl1 values(3001,3);
insert into test_tbl2 values(1,'name1');
insert into test_tbl2 values(2,'name2');
insert into test_tbl2 values(3,'name3');
3、查询表中数据,select t.* from test_tbl2 t ;
4、编写sql,两表通过pid与id关联;select t1.*, t2.* from test_tbl1 t1 join test_tbl2 t2 on t1.p_id = t2.id;
MySQL创建关联表是非常基础的知识,下面就为您举例说明MySQL创建关联表的方法,
MySQL创建关联表可以理解为是两个表之间有个外键关系,但这两个表必须满足三个条件1.两个表必须是InnoDB数据引擎2.使用在外键关系的域必须为索引型(Index)3.使用在外键关系的域必须与数据类型相似下面分别建两个表来说明一下:
Create TABLE IF NOT EXISTS `books` ( `book_id` smallint(6) NOT NULL auto_increment COMMENT '书籍编号', `book_name` char(20) NOT NULL COMMENT '书名', `book_pic` varchar(200) NOT NULL COMMENT '封面', `book_author` char(20) NOT NULL COMMENT '作者', `book_pub` char(40) NOT NULL COMMENT '出版社', `book_sort` char(6) NOT NULL COMMENT '分类', `book_owner` char(6) default NULL COMMENT '所有者', `book_borrower` char(7) default NULL COMMENT '借阅者', `book_borrower_time` date default NULL COMMENT '借阅时间', PRIMARY KEY (`book_id`), INDEX (book_borrower)) ENGINE=InnoDB CHARACTER SET utf8 COLLATE utf8_general_ci AUTO_INCREMENT=5 ;
Create TABLE IF NOT EXISTS `parts` ( `part_id` smallint(6) NOT NULL COMMENT '成员编号', `part_name` varchar(6) NOT NULL COMMENT '成员名', `part_mail` varchar(50) NOT NULL COMMENT '邮箱', `part_pass` varchar(20) NOT NULL COMMENT '密码', PRIMARY KEY (`part_id`), FOREIGN KEY(part_name) REFERENCES books(book_borrower) on delete cascade on update cascade) ENGINE=InnoDB CHARACTER SET utf8 COLLATE utf8_general_ci;分析一下books表和parts表,创建他们的关联,我用了books表的book_borrower字段 创建表时索引并选择InnoDB为表引擎。而parts表即part_name字段为外键,关联到books表的book_borrower字段.注意两 个字段分别是char和varchar都是字符串类型。on delete cascade意思为当books表有相关记录删除时,那parts表也会跟着删除相关联的记录. 理论上parts表的part_name字段也应该要创建索引才对,但实验证创建关联后自动索引了。
方法和操作步骤如下:
1、首先,创建一个测试表,如下图所示,然后进入下一步。
2、其次,插入测试数据,如下图所示,然后进入下一步。
3、接着,完成上述步骤后,查询表中的数据,“select t.* from test_tbl2 t ”,如下图所示,然后进入下一步。
4、最后,完成上述步骤后,编写sql,两个表通过pid与id关联, “select t1.*, t2.* from test_tbl1 t1 join test_tbl2 t2 on t1.p_id = t2.id;”,如下图所示。这样,问题就解决了。
多表关联UPDATE语句怎么写呀?
update A,B set A.sj1=B.sj2
where
a.hm1=b.hm1
新手,不知道这样写对不对。
求助:mysql的多表关联更新!!!
在数据库设计的时候,对于你上面这样的表,不应该再定义zongping字段,因为该字段的数值完全由其它字段简单计算得出,在磁盘上保存它的值完全是浪费空间。
为了搜索数据的方便,我们应该定义一个表,仅仅包含pingshi、qizhong、qimo三个字段,然后定义一个视图(VIEW),视图包含计算字段,假设表名为tb,那么视图vw的定义语句为:
CREATE
VIEW
vw
(pingshi,qizhong,qimo,zongping)
AS
SELECT
pingshi,qizhong,qimo,pingshi*0.1+qizhong*0.3+qimo*0.6
FROM
tb
这样以后可以对vw视图进行查询,其数据和tb表相同,只是对了计算字段。
MYSQL里面建立视图的语句为:
CREATE VIEW语法
CREATE [OR REPLACE] [ALGORITHM = {UNDEFINED | MERGE | TEMPTABLE}]
VIEW view_name [(column_list)]
AS select_statement
[WITH [CASCADED | LOCAL] CHECK OPTION]
该语句能创建新的视图,如果给定了OR REPLACE子句,该语句还能替换已有的视图。select_statement是一种SELECT语句,它给出了视图的定义。该语句可从基表或其他视图进行选择。
该语句要求具有针对视图的CREATE VIEW权限,以及针对由SELECT语句选择的每一列上的某些权限。对于在SELECT语句中其他地方使用的列,必须具有SELECT权限。如果还有OR REPLACE子句,必须在视图上具有DROP权限。
视图属于数据库。在默认情况下,将在当前数据库创建新视图。要想在给定数据库中明确创建视图,创建时,应将名称指定为db_name.view_name。
mysql> CREATE VIEW test.v AS SELECT * FROM t;
表和视图共享数据库中相同的名称空间,因此,数据库不能包含具有相同名称的表和视图。
视图必须具有唯一的列名,不得有重复,就像基表那样。默认情况下,由SELECT语句检索的列名将用作视图列名。要想为视图列定义明确的名称,可使用可选的column_list子句,列出由逗号隔开的ID。column_list中的名称数目必须等于SELECT语句检索的列数。
SELECT语句检索的列可以是对表列的简单引用。也可以是使用函数、常量值、操作符等的表达式。
对于SELECT语句中不合格的表或视图,将根据默认的数据库进行解释。通过用恰当的数据库名称限定表或视图名,视图能够引用表或其他数据库中的视图。
能够使用多种SELECT语句创建视图。视图能够引用基表或其他视图。它能使用联合、UNION和子查询。SELECT甚至不需引用任何表。在下面的示例中,定义了从另一表选择两列的视图,并给出了根据这些列计算的表达式:
mysql> CREATE TABLE t (qty INT, price INT);
mysql> INSERT INTO t VALUES(3, 50);
mysql> CREATE VIEW v AS SELECT qty, price, qty*price AS value FROM t;
mysql> SELECT * FROM v;
+------+-------+-------+
| qty | price | value |
+------+-------+-------+
| 3 | 50 | 150 |
+------+-------+-------+
视图定义服从下述限制:
· SELECT语句不能包含FROM子句中的子查询。
· SELECT语句不能引用系统或用户变量。
· SELECT语句不能引用预处理语句参数。
· 在存储子程序内,定义不能引用子程序参数或局部变量。
· 在定义中引用的表或视图必须存在。但是,创建了视图后,能够舍弃定义引用的表或视图。要想检查视图定义是否存在这类问题,可使用CHECK TABLE语句。
· 在定义中不能引用TEMPORARY表,不能创建TEMPORARY视图。
· 在视图定义中命名的表必须已存在。
· 不能将触发程序与视图关联在一起。
在视图定义中允许使用ORDER BY,但是,如果从特定视图进行了选择,而该视图使用了具有自己ORDER BY的语句,它将被忽略。
对于定义中的其他选项或子句,它们将被增加到引用视图的语句的选项或子句中,但效果未定义。例如,如果在视图定义中包含LIMIT子句,而且从特定视图进行了选择,而该视图使用了具有自己LIMIT子句的语句,那么对使用哪个LIMIT未作定义。相同的原理也适用于其他选项,如跟在SELECT关键字后的ALL、DISTINCT或SQL_SMALL_RESULT,并适用于其他子句,如INTO、FOR UPDATE、LOCK IN SHARE MODE、以及PROCEDURE。
如果创建了视图,并通过更改系统变量更改了查询处理环境,会影响从视图获得的结果:
mysql> CREATE VIEW v AS SELECT CHARSET(CHAR(65)), COLLATION(CHAR(65));
Query OK, 0 rows affected (0.00 sec)
mysql> SET NAMES 'latin1';
Query OK, 0 rows affected (0.00 sec)
mysql> SELECT * FROM v;
+-------------------+---------------------+
| CHARSET(CHAR(65)) | COLLATION(CHAR(65)) |
+-------------------+---------------------+
| latin1 | latin1_swedish_ci |
+-------------------+---------------------+
1 row in set (0.00 sec)
mysql> SET NAMES 'utf8';
Query OK, 0 rows affected (0.00 sec)
mysql> SELECT * FROM v;
+-------------------+---------------------+
| CHARSET(CHAR(65)) | COLLATION(CHAR(65)) |
+-------------------+---------------------+
| utf8 | utf8_general_ci |
+-------------------+---------------------+
1 row in set (0.00 sec)
可选的ALGORITHM子句是对标准SQL的MySQL扩展。ALGORITHM可取三个值:MERGE、TEMPTABLE或UNDEFINED。如果没有ALGORITHM子句,默认算法是UNDEFINED(未定义的)。算法会影响MySQL处理视图的方式。
对于MERGE,会将引用视图的语句的文本与视图定义合并起来,使得视图定义的某一部分取代语句的对应部分。
对于TEMPTABLE,视图的结果将被置于临时表中,然后使用它执行语句。
对于UNDEFINED,MySQL将选择所要使用的算法。如果可能,它倾向于MERGE而不是TEMPTABLE,这是因为MERGE通常更有效,而且如果使用了临时表,视图是不可更新的。
明确选择TEMPTABLE的1个原因在于,创建临时表之后、并在完成语句处理之前,能够释放基表上的锁定。与MERGE算法相比,锁定释放的速度更快,这样,使用视图的其他客户端不会被屏蔽过长时间。
视图算法可以是UNDEFINED,有三种方式:
· 在CREATE VIEW语句中没有ALGORITHM子句。
· CREATE VIEW语句有1个显式ALGORITHM = UNDEFINED子句。
· 为仅能用临时表处理的视图指定ALGORITHM = MERGE。在这种情况下,MySQL将生成告警,并将算法设置为UNDEFINED。
正如前面所介绍的那样,通过将视图定义中的对应部分合并到引用视图的语句中,对MERGE进行处理。在下面的示例中,简要介绍了MERGE的工作方式。在该示例中,假定有1个具有下述定义的视图v_merge:
CREATE ALGORITHM = MERGE VIEW v_merge (vc1, vc2) AS
SELECT c1, c2 FROM t WHERE c3 > 100;
示例1:假定发出了下述语句:
SELECT * FROM v_merge;
MySQL以下述方式处理语句:
· v_merge成为t
· *成为vc1、vc2,与c1、c2对应
· 增加视图WHERE子句
所产生的将执行的语句为:
SELECT c1, c2 FROM t WHERE c3 > 100;
示例2:假定发出了下述语句:
SELECT * FROM v_merge WHERE vc1 < 100;
该语句的处理方式与前面介绍的类似,但vc1 < 100变为c1 < 100,并使用AND连接词将视图的WHERE子句添加到语句的WHERE子句中(增加了圆括号以确保以正确的优先顺序执行子句部分)。所得的将要执行的语句变为:
SELECT c1, c2 FROM t WHERE (c3 > 100) AND (c1 < 100);
事实上,将要执行的语句是具有下述形式的WHERE子句:
WHERE (select WHERE) AND (view WHERE)
MERGE算法要求视图中的行和基表中的行具有一对一的关系。如果不具有该关系。必须使用临时表取而代之。如果视图包含下述结构中的任何一种,将失去一对一的关系:
· 聚合函数(SUM(), MIN(), MAX(), COUNT()等)。
· DISTINCT
· GROUP BY
· HAVING
· UNION或UNION ALL
· 仅引用文字值(在该情况下,没有基本表)。
某些视图是可更新的。也就是说,可以在诸如UPDATE、DELETE或INSERT等语句中使用它们,以更新基表的内容。对于可更新的视图,在视图中的行和基表中的行之间必须具有一对一的关系。还有一些特定的其他结构,这类结构会使得视图不可更新。更具体地讲,如果视图包含下述结构中的任何一种,那么它就是不可更新的:
· 聚合函数(SUM(), MIN(), MAX(), COUNT()等)。
· DISTINCT
· GROUP BY
· HAVING
· UNION或UNION ALL
· 位于选择列表中的子查询
· Join
· FROM子句中的不可更新视图
· WHERE子句中的子查询,引用FROM子句中的表。
· 仅引用文字值(在该情况下,没有要更新的基本表)。
· ALGORITHM = TEMPTABLE(使用临时表总会使视图成为不可更新的)。
关于可插入性(可用INSERT语句更新),如果它也满足关于视图列的下述额外要求,可更新的视图也是可插入的:
· 不得有重复的视图列名称。
· 视图必须包含没有默认值的基表中的所有列。
· 视图列必须是简单的列引用而不是导出列。导出列不是简单的列引用,而是从表达式导出的。下面给出了一些导出列示例:
· 3.14159
· col1 + 3
· UPPER(col2)
· col3 / col4
· (subquery)
混合了简单列引用和导出列的视图是不可插入的,但是,如果仅更新非导出列,视图是可更新的。考虑下述视图:
CREATE VIEW v AS SELECT col1, 1 AS col2 FROM t;
该视图是不可插入的,这是因为col2是从表达式导出的。但是,如果更新时不更新col2,它是可更新的。这类更新是允许的:
UPDATE v SET col1 = 0;
下述更新是不允许的,原因在于,它试图更新导出列:
UPDATE v SET col2 = 0;
在某些情况下,能够更新多表视图,假定它能使用MERGE算法进行处理。为此,视图必须使用内部联合(而不是外部联合或UNION)。此外,仅能更新视图定义中的单个表,因此,SET子句必须仅命名视图中某一表的列。即使从理论上讲也是可更新的,不允许使用UNION ALL的视图,这是因为,在实施中将使用临时表来处理它们。
对于多表可更新视图,如果是将其插入单个表中,INSERT能够工作。不支持DELETE。
对于可更新视图,可给定WITH CHECK OPTION子句来防止插入或更新行,除非作用在行上的select_statement中的WHERE子句为“真”。
在关于可更新视图的WITH CHECK OPTION子句中,当视图是根据另一个视图定义的时,LOCAL和CASCADED关键字决定了检查测试的范围。LOCAL关键字对CHECK OPTION进行了限制,使其仅作用在定义的视图上,CASCADED会对将进行评估的基表进行检查。如果未给定任一关键字,默认值为CASCADED。请考虑下述表和视图集合的定义:
mysql> CREATE TABLE t1 (a INT);
mysql> CREATE VIEW v1 AS SELECT * FROM t1 WHERE a < 2
-> WITH CHECK OPTION;
mysql> CREATE VIEW v2 AS SELECT * FROM v1 WHERE a > 0
-> WITH LOCAL CHECK OPTION;
mysql> CREATE VIEW v3 AS SELECT * FROM v1 WHERE a > 0
-> WITH CASCADED CHECK OPTION;
这里,视图v2和v3是根据另一视图v1定义的。v2具有LOCAL检查选项,因此,仅会针对v2检查对插入项进行测试。v3具有CASCADED检查选项,因此,不仅会针对它自己的检查对插入项进行测试,也会针对基本视图的检查对插入项进行测试。在下面的语句中,介绍了这些差异:
ql> INSERT INTO v2 VALUES (2);
Query OK, 1 row affected (0.00 sec)
mysql> INSERT INTO v3 VALUES (2);
ERROR 1369 (HY000): CHECK OPTION failed 'test.v3'
视图的可更新性可能会受到系统变量updatable_views_with_limit的值的影响。请参见5.3.3节,“服务器系统变量”。
INFORMATION_SCHEMA包含1个VIEWS表,从该表可获取关于视图对象的信息。请参见23.1.15节,“INFORMATION_SCHEMA VIEWS表”。
在数据库设计的时候,对于你上面这样的表,不应该再定义zongping字段,因为该字段的数值完全由其它字段简单计算得出,在磁盘上保存它的值完全是浪费空间。
为了搜索数据的方便,我们应该定义一个表,仅仅包含pingshi、qizhong、qimo三个字段,然后定义一个视图(VIEW),视图包含计算字段,假设表名为tb,那么视图vw的定义语句为:
CREATE VIEW vw (pingshi,qizhong,qimo,zongping)
AS SELECT pingshi,qizhong,qimo,pingshi*0.1+qizhong*0.3+qimo*0.6 FROM tb
这样以后可以对vw视图进行查询,其数据和tb表相同,只是对了计算字段。
MYSQL里面建立视图的语句为:
CREATE?VIEW语法
CREATE?[OR?REPLACE]?[ALGORITHM?=?{UNDEFINED?|?MERGE?|?TEMPTABLE}]
????VIEW?view_name?[(column_list)]
????AS?select_statement
????[WITH?[CASCADED?|?LOCAL]?CHECK?OPTION]
该语句能创建新的视图,如果给定了OR?REPLACE子句,该语句还能替换已有的视图。select_statement是一种SELECT语句,它给出了视图的定义。该语句可从基表或其他视图进行选择。
该语句要求具有针对视图的CREATE?VIEW权限,以及针对由SELECT语句选择的每一列上的某些权限。对于在SELECT语句中其他地方使用的列,必须具有SELECT权限。如果还有OR?REPLACE子句,必须在视图上具有DROP权限。
视图属于数据库。在默认情况下,将在当前数据库创建新视图。要想在给定数据库中明确创建视图,创建时,应将名称指定为db_name.view_name。
mysql>?CREATE?VIEW?test.v?AS?SELECT?*?FROM?t;
表和视图共享数据库中相同的名称空间,因此,数据库不能包含具有相同名称的表和视图。
视图必须具有唯一的列名,不得有重复,就像基表那样。默认情况下,由SELECT语句检索的列名将用作视图列名。要想为视图列定义明确的名称,可使用可选的column_list子句,列出由逗号隔开的ID。column_list中的名称数目必须等于SELECT语句检索的列数。
SELECT语句检索的列可以是对表列的简单引用。也可以是使用函数、常量值、操作符等的表达式。
对于SELECT语句中不合格的表或视图,将根据默认的数据库进行解释。通过用恰当的数据库名称限定表或视图名,视图能够引用表或其他数据库中的视图。
能够使用多种SELECT语句创建视图。视图能够引用基表或其他视图。它能使用联合、UNION和子查询。SELECT甚至不需引用任何表。在下面的示例中,定义了从另一表选择两列的视图,并给出了根据这些列计算的表达式:
mysql>?CREATE?TABLE?t?(qty?INT,?price?INT);
mysql>?INSERT?INTO?t?VALUES(3,?50);
mysql>?CREATE?VIEW?v?AS?SELECT?qty,?price,?qty*price?AS?value?FROM?t;
mysql>?SELECT?*?FROM?v;
+------+-------+-------+
|?qty??|?price?|?value?|
+------+-------+-------+
|????3?|????50?|???150?|
+------+-------+-------+
视图定义服从下述限制:
·?????????SELECT语句不能包含FROM子句中的子查询。
·?????????SELECT语句不能引用系统或用户变量。
·?????????SELECT语句不能引用预处理语句参数。
·?????????在存储子程序内,定义不能引用子程序参数或局部变量。
·?????????在定义中引用的表或视图必须存在。但是,创建了视图后,能够舍弃定义引用的表或视图。要想检查视图定义是否存在这类问题,可使用CHECK?TABLE语句。
·?????????在定义中不能引用TEMPORARY表,不能创建TEMPORARY视图。
·?????????在视图定义中命名的表必须已存在。
·?????????不能将触发程序与视图关联在一起。
在视图定义中允许使用ORDER?BY,但是,如果从特定视图进行了选择,而该视图使用了具有自己ORDER?BY的语句,它将被忽略。
对于定义中的其他选项或子句,它们将被增加到引用视图的语句的选项或子句中,但效果未定义。例如,如果在视图定义中包含LIMIT子句,而且从特定视图进行了选择,而该视图使用了具有自己LIMIT子句的语句,那么对使用哪个LIMIT未作定义。相同的原理也适用于其他选项,如跟在SELECT关键字后的ALL、DISTINCT或SQL_SMALL_RESULT,并适用于其他子句,如INTO、FOR?UPDATE、LOCK?IN?SHARE?MODE、以及PROCEDURE。
如果创建了视图,并通过更改系统变量更改了查询处理环境,会影响从视图获得的结果:
mysql>?CREATE?VIEW?v?AS?SELECT?CHARSET(CHAR(65)),?COLLATION(CHAR(65));
Query?OK,?0?rows?affected?(0.00?sec)
?
mysql>?SET?NAMES?'latin1';
Query?OK,?0?rows?affected?(0.00?sec)
?
mysql>?SELECT?*?FROM?v;
+-------------------+---------------------+
|?CHARSET(CHAR(65))?|?COLLATION(CHAR(65))?|
+-------------------+---------------------+
|?latin1????????????|?latin1_swedish_ci???|
+-------------------+---------------------+
1?row?in?set?(0.00?sec)
?
mysql>?SET?NAMES?'utf8';
Query?OK,?0?rows?affected?(0.00?sec)
?
mysql>?SELECT?*?FROM?v;
+-------------------+---------------------+
|?CHARSET(CHAR(65))?|?COLLATION(CHAR(65))?|
+-------------------+---------------------+
|?utf8??????????????|?utf8_general_ci?????|
+-------------------+---------------------+
1?row?in?set?(0.00?sec)
可选的ALGORITHM子句是对标准SQL的MySQL扩展。ALGORITHM可取三个值:MERGE、TEMPTABLE或UNDEFINED。如果没有ALGORITHM子句,默认算法是UNDEFINED(未定义的)。算法会影响MySQL处理视图的方式。
对于MERGE,会将引用视图的语句的文本与视图定义合并起来,使得视图定义的某一部分取代语句的对应部分。
对于TEMPTABLE,视图的结果将被置于临时表中,然后使用它执行语句。
对于UNDEFINED,MySQL将选择所要使用的算法。如果可能,它倾向于MERGE而不是TEMPTABLE,这是因为MERGE通常更有效,而且如果使用了临时表,视图是不可更新的。
明确选择TEMPTABLE的1个原因在于,创建临时表之后、并在完成语句处理之前,能够释放基表上的锁定。与MERGE算法相比,锁定释放的速度更快,这样,使用视图的其他客户端不会被屏蔽过长时间。
视图算法可以是UNDEFINED,有三种方式:
·?????????在CREATE?VIEW语句中没有ALGORITHM子句。
·?????????CREATE?VIEW语句有1个显式ALGORITHM?=?UNDEFINED子句。
·?????????为仅能用临时表处理的视图指定ALGORITHM?=?MERGE。在这种情况下,MySQL将生成告警,并将算法设置为UNDEFINED。
正如前面所介绍的那样,通过将视图定义中的对应部分合并到引用视图的语句中,对MERGE进行处理。在下面的示例中,简要介绍了MERGE的工作方式。在该示例中,假定有1个具有下述定义的视图v_merge:
CREATE?ALGORITHM?=?MERGE?VIEW?v_merge?(vc1,?vc2)?AS
SELECT?c1,?c2?FROM?t?WHERE?c3?>?100;
示例1:假定发出了下述语句:
SELECT?*?FROM?v_merge;
MySQL以下述方式处理语句:
·?????????v_merge成为t
·?????????*成为vc1、vc2,与c1、c2对应
·?????????增加视图WHERE子句
所产生的将执行的语句为:
SELECT?c1,?c2?FROM?t?WHERE?c3?>?100;
示例2:假定发出了下述语句:
SELECT?*?FROM?v_merge?WHERE?vc1?
<!--?100;该语句的处理方式与前面介绍的类似,但vc1?
<!--?100变为c1?<?100,并使用AND连接词将视图的WHERE子句添加到语句的WHERE子句中(增加了圆括号以确保以正确的优先顺序执行子句部分)。所得的将要执行的语句变为:SELECT?c1,?c2?FROM?t?WHERE?(c3?>?100)?AND?(c1?
<!--?100);事实上,将要执行的语句是具有下述形式的WHERE子句:
WHERE?(select?WHERE)?AND?(view?WHERE)
MERGE算法要求视图中的行和基表中的行具有一对一的关系。如果不具有该关系。必须使用临时表取而代之。如果视图包含下述结构中的任何一种,将失去一对一的关系:
·?????????聚合函数(SUM(),?MIN(),?MAX(),?COUNT()等)。
·?????????DISTINCT?
·?????????GROUP?BY?
·?????????HAVING?
·?????????UNION或UNION?ALL?
·?????????仅引用文字值(在该情况下,没有基本表)。
某些视图是可更新的。也就是说,可以在诸如UPDATE、DELETE或INSERT等语句中使用它们,以更新基表的内容。对于可更新的视图,在视图中的行和基表中的行之间必须具有一对一的关系。还有一些特定的其他结构,这类结构会使得视图不可更新。更具体地讲,如果视图包含下述结构中的任何一种,那么它就是不可更新的:
·?????????聚合函数(SUM(),?MIN(),?MAX(),?COUNT()等)。
·?????????DISTINCT?
·?????????GROUP?BY?
·?????????HAVING?
·?????????UNION或UNION?ALL?
·?????????位于选择列表中的子查询
·?????????Join?
·?????????FROM子句中的不可更新视图
·?????????WHERE子句中的子查询,引用FROM子句中的表。
·?????????仅引用文字值(在该情况下,没有要更新的基本表)。
·?????????ALGORITHM?=?TEMPTABLE(使用临时表总会使视图成为不可更新的)。
关于可插入性(可用INSERT语句更新),如果它也满足关于视图列的下述额外要求,可更新的视图也是可插入的:
·?????????不得有重复的视图列名称。
·?????????视图必须包含没有默认值的基表中的所有列。
·?????????视图列必须是简单的列引用而不是导出列。导出列不是简单的列引用,而是从表达式导出的。下面给出了一些导出列示例:
·????????????????3.14159
·????????????????col1?+?3
·????????????????UPPER(col2)
·????????????????col3?/?col4
·????????????????(subquery)
混合了简单列引用和导出列的视图是不可插入的,但是,如果仅更新非导出列,视图是可更新的。考虑下述视图:
CREATE?VIEW?v?AS?SELECT?col1,?1?AS?col2?FROM?t;
该视图是不可插入的,这是因为col2是从表达式导出的。但是,如果更新时不更新col2,它是可更新的。这类更新是允许的:
UPDATE?v?SET?col1?=?0;
下述更新是不允许的,原因在于,它试图更新导出列:
UPDATE?v?SET?col2?=?0;
在某些情况下,能够更新多表视图,假定它能使用MERGE算法进行处理。为此,视图必须使用内部联合(而不是外部联合或UNION)。此外,仅能更新视图定义中的单个表,因此,SET子句必须仅命名视图中某一表的列。即使从理论上讲也是可更新的,不允许使用UNION?ALL的视图,这是因为,在实施中将使用临时表来处理它们。
对于多表可更新视图,如果是将其插入单个表中,INSERT能够工作。不支持DELETE。
对于可更新视图,可给定WITH?CHECK?OPTION子句来防止插入或更新行,除非作用在行上的select_statement中的WHERE子句为“真”。
在关于可更新视图的WITH?CHECK?OPTION子句中,当视图是根据另一个视图定义的时,LOCAL和CASCADED关键字决定了检查测试的范围。LOCAL关键字对CHECK?OPTION进行了限制,使其仅作用在定义的视图上,CASCADED会对将进行评估的基表进行检查。如果未给定任一关键字,默认值为CASCADED。请考虑下述表和视图集合的定义:
mysql>?CREATE?TABLE?t1?(a?INT);
mysql>?CREATE?VIEW?v1?AS?SELECT?*?FROM?t1?WHERE?a?
<!--?2????->?WITH?CHECK?OPTION;
mysql>?CREATE?VIEW?v2?AS?SELECT?*?FROM?v1?WHERE?a?>?0
????->?WITH?LOCAL?CHECK?OPTION;
mysql>?CREATE?VIEW?v3?AS?SELECT?*?FROM?v1?WHERE?a?>?0
????->?WITH?CASCADED?CHECK?OPTION;
这里,视图v2和v3是根据另一视图v1定义的。v2具有LOCAL检查选项,因此,仅会针对v2检查对插入项进行测试。v3具有CASCADED检查选项,因此,不仅会针对它自己的检查对插入项进行测试,也会针对基本视图的检查对插入项进行测试。在下面的语句中,介绍了这些差异:
ql>?INSERT?INTO?v2?VALUES?(2);
Query?OK,?1?row?affected?(0.00?sec)
mysql>?INSERT?INTO?v3?VALUES?(2);
ERROR?1369?(HY000):?CHECK?OPTION?failed?'test.v3'
视图的可更新性可能会受到系统变量updatable_views_with_limit的值的影响。请参见5.3.3节,“服务器系统变量”。
INFORMATION_SCHEMA包含1个VIEWS表,从该表可获取关于视图对象的信息。请参见23.1.15节,“INFORMATION_SCHEMA?VIEWS表”。
update语句
UPDATE语句如下:
单表的MySQL UPDATE语句:
UPDATE [LOW_PRIORITY] [IGNORE] tbl_name SET col_name1=expr1 [, col_name2=expr2 ...] [WHERE where_definition] [ORDER BY ...] [LIMIT row_count]
多表的UPDATE语句:
UPDATE [LOW_PRIORITY] [IGNORE] table_references SET col_name1=expr1 [, col_name2=expr2 ...] [WHERE where_definition]
UPDATE语句支持以下修饰符:
如果您使用LOW_PRIORITY关键词,则UPDATE的执行被延迟了,直到没有其它的客户端从表中读取为止。
如果您使用IGNORE关键词,则即使在更新过程中出现错误,更新语句也不会中断。如果出现了重复关键字冲突,则这些行不会被更新。如果列被更新后,新值会导致数据转化错误,则这些行被更新为最接近的合法的值。
mysql中的update语句怎么写
首先,单表的UPDATE语句:
UPDATE [LOW_PRIORITY] [IGNORE] tbl_name
SET col_name1=expr1 [, col_name2=expr2 ...]
[WHERE where_definition]
[ORDER BY ...]
[LIMIT row_count]
其次,多表的UPDATE语句:
UPDATE [LOW_PRIORITY] [IGNORE] table_references
SET col_name1=expr1 [, col_name2=expr2 ...]
[WHERE where_definition]
update语句作为mysql更新语句,set后面紧接着的是需要更新的列明和想要更新的值where后面限定更新的条件,order by根据某一个字段排序后,将会由排序后由上到下逐条更新,limit将会限制更新的条数.
update 表名 set 字段名=字段值 where 条件
如 update a set name='xiaoming' where name='';
如果是多表查询
update 表1 a inner join 表2 b on ab表的关联 set a.字段=b.字段
如 update table1 a inner join table2 b on a.id=b.id set a.name=b.name
就是在table1表和table2表id相同时 把table2的name值赋给table1的name
update tablename set column=123,column2=234 where 条件
SQL UPDATE 命令如果我们需要修改或更新 MySQL 中的数据,我们可以使用 SQL UPDATE 命令来操作。
具体语法参考:
from 树懒学堂 - 一站式数据知识平台
注意:
你可以同时更新一个或多个字段。
你可以在 WHERE 子句中指定任何条件。
你可以在一个单独表中同时更新数据。
当你需要更新数据表中指定行的数据时 WHERE 子句是非常有用的。
update后可接两张表吗,
一条update只能修改一张表里的字段,但是可以关联多张表去修改。不知道你用的是什么数据库。
常用的sqlserver格式如下:
update table1 set a.字段1=b.字段1,....,a.字段N=b.字段N from table1 a,table2 b where 两个表的关联字段。
常用的oracle格式如下:
update table1 a set (a.字段1,....,a.字段N) =(select b.字段1,...,b.字段N from table2 b where 两个表的关联字段) where exists (select 1 from table2 b where 两个表的关联字段)。
注意oracle语句里的exists不能省略,否则会导致没有对应关系的数据修改错误,甚至会报错。
可以,如:
update 表A set 字段A=tba.字段B +tba.字段C
from
(select * from 表B as a left join 表C as b on a.主键=b.主键) as tba
where tba.主键=表A.主键
不能update两张表,但是可以出现在where条件中
如果想用一张表的字段值更新另一张表时。需要注意
mysql后面可以跟多张表,包括临时表,如:
update apple app,bus bus set app.structid=bus.id
where app.strucname=bus.strucname
oracle不可以跟多张,可以配合exists来解决。(不怕麻烦的可以用存储过程)
一条update只能修改一张表里的字段,但是可以关联多张表去修改。不知道你用的是什么数据库。常用的sqlserver格式如下:
update table1 set a.字段1=b.字段1,....,a.字段N=b.字段N from table1 a,table2 b where 两个表的关联字段。
常用的oracle格式如下:
update table1 a set (a.字段1,....,a.字段N) =(select b.字段1,...,b.字段N from table2 b where 两个表的关联字段) where exists (select 1 from table2 b where 两个表的关联字段)。
注意oracle语句里的exists不能省略,否则会导致没有对应关系的数据修改错误,甚至会报错。
EXISTS 指定一个子查询,检测 行 的存在。
检测select 1 from yls_test c where a.cert_no =c.cert_no 是否存在如果存在执行update 如果不存在,语句就成了 update month_apply a set a.emp_name = (select name from yls_test b where a.cert_no = b.cert_no) 这样的语句是不执行更新命令的。
mysql多表关联更新问题
按照描述这么写:
update 表名 t1set t1.name=(select name from 表名 t2where t2.pid=t1.pid and t2.name is not null and rownum=1)where t1.name is null;