用一道题 来 复习 MySQL 的 复杂 sql 语句
1.前言
太久没有在数据库做一些复杂的sql了,基本上将数据库的查询逻辑全放在了Java里做,
一来呢,可以减轻数据库的负担,二来呢,在java写,逻辑感会更强,数据类型更丰富也容易操作。
然而。。。面试却喜欢靠复杂的sql ,好吧,即便我不想,但复习一波还是免不了的。
常用的关系型数据库有 MySQL和Oracle 。Oracle 比较喜欢使用存储过程做业务 ,当然,MySQL也可以,但是没怎么用,
自从工程使用mybatis框架,就不再使用存储过程了,业务基本是增删改查,查询数据的逻辑都是从数据库取相应数据出来后用Java计算,
再从数据库获取最终想要的数据,本来是本着减轻数据库负担才这样做的,并发操作会用上积极锁【乐观锁】,因此也就不需要担心 脏数据问题。
MySQL和Oracle的语法部分是不同的,有时候用着MySQL,写着写着就用上了Oracle的语法,还一脸懵逼的查了半天到底哪里错,不常使用的东西就是容易忘。
总结:
(1)Oracle 使用nvl() 函数,MySQL使用 ifnull() 函数 来对数据进行判断是否为空,
如果是空则使用替代的数据 ,参数一样 ,如if(x.age,0),意思是如果年龄字段为空则
输出 0 .
(2)sum()函数是运算函数,允许 加减乘除计算 ,如果要使用,则必须使用
group by 分组 ,限定好分组 sum获取的计算数据才不会错,否则将会导致全表计算在一起。
(3)avg()函数是计算平均数的,用法根据需要与 group by 分组配合使用,
如果是计算全表某字段的平均分,则不要使用。
(4)having 关键字可以筛选分组后的各组数据,也就是说可对分组完成后的数据做逻辑条件判断 ,与where类似,但是where无法这样使用,因为where关键字无法与聚合函数一起使用
2.复习题
数据库源码

/*
Navicat MySQL Data Transfer Source Server : cen
Source Server Version : 50528
Source Host : localhost:3306
Source Database : kktest Target Server Type : MYSQL
Target Server Version : 50528
File Encoding : 65001 Date: 2020-06-17 08:21:15
*/ SET FOREIGN_KEY_CHECKS=0; -- ----------------------------
-- Table structure for bjb
-- ----------------------------
DROP TABLE IF EXISTS `bjb`;
CREATE TABLE `bjb` (
`id` int(11) NOT NULL AUTO_INCREMENT,
`name` varchar(20) DEFAULT NULL,
PRIMARY KEY (`id`)
) ENGINE=InnoDB AUTO_INCREMENT=5 DEFAULT CHARSET=utf8; -- ----------------------------
-- Records of bjb
-- ----------------------------
INSERT INTO `bjb` VALUES ('1', '一班');
INSERT INTO `bjb` VALUES ('2', '2班');
INSERT INTO `bjb` VALUES ('3', '3班'); -- ----------------------------
-- Table structure for cjb
-- ----------------------------
DROP TABLE IF EXISTS `cjb`;
CREATE TABLE `cjb` (
`id` int(11) NOT NULL AUTO_INCREMENT,
`id_sx` int(11) DEFAULT NULL,
`yw` int(11) DEFAULT NULL,
`sx` int(11) DEFAULT NULL,
PRIMARY KEY (`id`)
) ENGINE=InnoDB AUTO_INCREMENT=11 DEFAULT CHARSET=utf8; -- ----------------------------
-- Records of cjb
-- ----------------------------
INSERT INTO `cjb` VALUES ('1', '1', '77', '67');
INSERT INTO `cjb` VALUES ('2', '2', '32', '27');
INSERT INTO `cjb` VALUES ('3', '3', '98', '78');
INSERT INTO `cjb` VALUES ('4', '4', '68', '63');
INSERT INTO `cjb` VALUES ('5', '5', '66', '77');
INSERT INTO `cjb` VALUES ('6', '6', '99', '88');
INSERT INTO `cjb` VALUES ('7', '7', '75', '45');
INSERT INTO `cjb` VALUES ('8', '8', '77', '88');
INSERT INTO `cjb` VALUES ('9', '9', '65', '81');
INSERT INTO `cjb` VALUES ('10', '10', '83', '89'); -- ----------------------------
-- Table structure for xsb
-- ----------------------------
DROP TABLE IF EXISTS `xsb`;
CREATE TABLE `xsb` (
`id` int(11) NOT NULL AUTO_INCREMENT,
`name` varchar(20) DEFAULT NULL,
`id_banji` int(11) DEFAULT NULL,
PRIMARY KEY (`id`)
) ENGINE=InnoDB AUTO_INCREMENT=12 DEFAULT CHARSET=utf8; -- ----------------------------
-- Records of xsb
-- ----------------------------
INSERT INTO `xsb` VALUES ('1', '岑', '1');
INSERT INTO `xsb` VALUES ('2', 'cen', '1');
INSERT INTO `xsb` VALUES ('3', 'y', '2');
INSERT INTO `xsb` VALUES ('4', 'u', '3');
INSERT INTO `xsb` VALUES ('5', 'yue', '2');
INSERT INTO `xsb` VALUES ('6', 'kk', '2');
INSERT INTO `xsb` VALUES ('7', 'tom', '1');
INSERT INTO `xsb` VALUES ('8', 'lili', '1');
INSERT INTO `xsb` VALUES ('9', 'kile', '3');
INSERT INTO `xsb` VALUES ('10', 'jack', '2');
INSERT INTO `xsb` VALUES ('11', 'hh', '2'); -- ----------------------------
-- Procedure structure for sp_add3
-- ----------------------------
DROP PROCEDURE IF EXISTS `sp_add3`;
DELIMITER ;;
CREATE DEFINER=`root`@`localhost` PROCEDURE `sp_add3`(a int, b int,out c int)
begin
set c=a+ b;
end
;;
DELIMITER ;
kktest.sql
学生表【字段意思:学生id、姓名、班级id】

班级表【字段意思:班级id、班级名称】

成绩表【字段意思:成绩id、学生id、语文成绩、数学成绩】

【注意:11号同学hh ,他没有成绩,他作弊被取消了考试资格,因此成绩表没有他的信息】
(1)查询所有学生的信息
写法一:
select x.id,x.name,b.name n from xsb x
left join bjb b on b.id = x.id_banji;
查询结果

写法二:
select x.id,x.name,b.name n2 from xsb x,bjb b
where b.id = x.id_banji;
查询结果与上图一样
(2)查询所有人的课程分数
写法一:【查询11号同学为null】
select x.name,b.name n ,c.yw ,c.sx from xsb x
left join bjb b on b.id = x.id_banji
left join cjb c on c.id_sx = x.id ;
查询结果

写法二:【查询11号同学为0】
select x.name,b.name n , ifnull(c.yw,0) , ifnull(c.sx,0) from xsb x
left join bjb b on b.id = x.id_banji
left join cjb c on c.id_sx = x.id ;
查询结果

写法三:【查询无11号同学】不使用left join会导致没有成绩的那个同学不显示,因为直接连表查询只会保留所有关联条件成立的数据
select x.name,b.name n , ifnull(c.yw,0) , ifnull(c.sx,0) from xsb x,bjb b,cjb c
WHERE b.id = x.id_banji
and c.id_sx = x.id ;
查询结果

(3)查询语文分数比“yue”的高的学生,【 如果是查询比“yue”的低, 不使用ifnull那么没有成绩的同学无法查看到】
select x.name,b.name n , ifnull(c.yw,0) from xsb x
left join bjb b on b.id = x.id_banji
left join cjb c on c.id_sx = x.id
where ifnull(c.yw,0) >
(
select c.yw from cjb c
left join xsb x on c.id_sx = x.id
where x.name= "yue"
)
查询结果

(4)查询各科都合格【分数>=60分】的学生(姓名、语文分数、数学分数)
select x.name , c.yw ,c.sx
from xsb x
left join cjb c on c.id_sx =x.id
WHERE c.yw>60 and c.sx >60
打印结果

(5)查询总分数(语文+数学)>=150的学生信息(姓名、班级名称、总分数)
select x.name,b.name n ,
ifnull(c.yw,0) as "语文",ifnull(c.sx,0) as "数学",
#sum是运算函数 ,在里面可以做加减乘除
sum(ifnull(c.yw,0) + ifnull(c.sx,0)) as "总分"
from xsb x
left join bjb b on b.id = x.id_banji
left join cjb c on c.id_sx = x.id
where (ifnull(c.yw,0) +ifnull(c.sx,0)) >=150
#计算总分必须要分组,加上这个GROUP BY x.id,表示以一位学生为一组计算总分,否则会全部加在一起
GROUP BY x.id
查询结果

(6)查询没有参加考试【没有成绩表】的学生(姓名、班级名称)
写法一:
select x.name,b.name n from xsb x
left join bjb b on b.id = x.id_banji
left join cjb c on c.id_sx = x.id
where x.id not in (select id_sx from cjb);
查询结果

写法二:
select x.name,b.name n from xsb x ,bjb b ,cjb c
WHERE x.id_banji = b.id and x.id not in (select id_sx from cjb)
GROUP BY x.name,b.name ;
查询结果与上图一样
(7)假设分数>=60分合格,分析学生的成绩是否合格
select x.name,if(c.yw>=60,"合格","不合格") as "语文成绩" ,
if(c.sx>=60,"合格","不合格") as "数学成绩"
from xsb x
left join cjb c on c.id_sx = x.id
查询结果

(8)查询有挂科【分数<60分】现象的学生(姓名、语文分数、数学分数)
select x.name ,ifnull( c.yw ,0),ifnull(c.sx,0)
from xsb x
left join cjb c on c.id_sx =x.id
WHERE ifnull( c.yw ,0) <60 or ifnull(c.sx,0)<60;
查询结果

(9)查询所有班级的平均分数(班级编号、班级名称、语文平均分数、数学平均分数)
写法一:
select b.name , AVG(c.yw) as "语文平均分数" ,AVG(c.sx) as "数学平均分数" from xsb x
left join bjb b on b.id = x.id_banji
left join cjb c on c.id_sx = x.id
GROUP BY b.id ;
查询结果

写法二:【主从表换了没影响】
select b.name , AVG(c.yw) as "语文平均分数" ,AVG(c.sx) as "数学平均分数" from bjb b
left join xsb x on b.id = x.id_banji
left join cjb c on c.id_sx = x.id
GROUP BY b.id
查询结果与上图一样
(10)查询班级人数>=3的班级(班级编号、班级名称、人数)
select b.id ,b.name , count(x.id) as "人数"
from bjb b
left join xsb x on b.id = x.id_banji
#HAVING 子句可以让我们筛选分组后的各组数据。
group by b.id having count(x.id) >=3
查询结果

用一道题 来 复习 MySQL 的 复杂 sql 语句的更多相关文章
- MySQL 常用的sql语句小结(待续)
mysql 常用的sql语句 1.查看数据库各个表中的记录数 USE information_schema; SELECT table_name,table_rows FROM tables WHER ...
- mysql使用基础 sql语句(一)
csdn博文地址:mysql使用基础 sql语句(一) 点击进入 命令行输入mysql -u root -p,回车再输入密码,进入mysql. 终端命令以分号作为一条语句的结束,可分为多行输入,只需 ...
- 监控mysql执行的sql语句
linux平台 监控mysql执行的sql语句 为了做好配合开发做性能和功能测试,方便监控正在执行的sql语句, 可以在/etc/mysqld中添加如下: log =/usr/local/mys ...
- MySQL的常用SQL语句.md
修改密码 这是常见的大家一般都要用的 首先 安装成功了打开cmd --> mysql -u root -p -->输入你的密码 修改mysql root用户密码 格式 ...
- mysql统计类似SQL语句查询次数
mysql统计类似SQL语句查询次数 vc-mysql-sniffer 工具抓取的sql分析. 1.先用shell脚本把所有enter符号替换为null,再根据语句前后的字符分隔语句 grep -Ev ...
- MySQL中执行sql语句错误 Error Code: 1093. You can't specify target table 'car' for update in FROM clause
MySQL中执行sql语句错误 Error Code: 1093. You can't specify target table 'car' for update in FROM clause 201 ...
- Oracle,SQL Server 数据库较MySql数据库,Sql语句差异
原文:Oracle,SQL Server 数据库较MySql数据库,Sql语句差异 Oracle,SQL Server 数据库较MySql数据库,Sql语句差异 1.关系型数据库 百度百科 关系数据库 ...
- 浅谈MySQL中优化sql语句查询常用的30种方法 - 转载
浅谈MySQL中优化sql语句查询常用的30种方法 1.对查询进行优化,应尽量避免全表扫描,首先应考虑在 where 及 order by 涉及的列上建立索引. 2.应尽量避免在 where 子句中使 ...
- mysql(数据库,sql语句,普通查询)
第1章 数据库 1.1 数据库概述 l 什么是数据库 数据库就是存储数据的仓库,其本质是一个文件系统,数据按照特定的格式将数据存储起来,用户可以对数据库中的数据进行增加,修改,删除及查询操作. l 什 ...
随机推荐
- 【C/C++】引用&的含义/语法/作为函数参数/函数返回值/本质/常量引用
含义 引用不产生副本,只是给原变量起了别名. 对引用变量的操作就是对原变量的操作. 基本语法 数据类型 &别名 = 原名 e.g. int a = 10; int &b = a; // ...
- Centos 常用指令
1.*.tar 用 tar xvf 解压 2.*.gz 用 gzip d或者gunzip 解压 3.*.tar.gz和*.tgz 用 tar xzf 解压 4.*.bz2 用 bzip2 d或者用 ...
- ASP.NET Core中使用漏桶算法限流
漏桶算法是限流的四大主流算法之一,其应用场景各种资料中介绍的不多,一般都是说应用在网络流量控制中.这里举两个例子: 1.目前家庭上网都会限制一个固定的带宽,比如100M.200M等,一栋楼有很多的用户 ...
- PostgreSql数据库安全加固
1.确保通过"主机" TCP / IP套接字登录已正确配置 描述 大量的身份验证方法可用于使用 TCP / IP套接字,包括: ?信任 ? 拒绝 ?md5 ?scram-sha-2 ...
- shell脚本 系统信息检测
一.简介 源码地址 日期:2018/4/12 介绍:根据指令展示不同的系统数据 效果图: 二.使用 适用:centos6+ 语言:中文 注意:无 下载 wget https://raw.githubu ...
- 转:StoryBoard快速上手
由于最近才接触到IOS,苹果已经建议storyboard来搭建所有界面了,于是我也追随时尚,直接开始使用storyboard.(不料在涉及到页 面跳转的时候,遇到的问题是:点击后没有任何反应)众所周知 ...
- BUUCFT pwn asis2016_b00ks
看师傅们wp的时候,我才知道这个道题是wiki上面的例题.我看了一些师傅的wp,发现大家都是一种做法,都是通过mmap堆地址,来找libc基地址的.而我试了一下fastbisn attack,发现也可 ...
- PDF补丁丁将发布开放源代码的1.0版本
近况 一个月前的今天,母亲永远离开了我. 想起四个月前,我送她了去住院.入院后,做了检查.检查结果没出,我的生日就到了.母亲很关心我的生日.在电话里,她祝我身体健康,又问媳妇有没有给我做生日餐桌的菜肴 ...
- Linux 三剑客之sed
目录 Linux 三剑客之sed 命令补充: sort命令 uniq命令 cut命令 tr命令 wc命令 三剑客 - sed 编辑模式: 定位分类: 实例如下: d模式--删除模式 p模式--打印 a ...
- Linux下使用JDK11部署Nacos启动报错:Could not find or load main class
Linux下使用JDK11部署Nacos 错误日志 /nacos/jdk-11.0.12/bin/java -server -Xms2g -Xmx2g -Xmn1g -XX:MetaspaceSize ...