搜索
您的当前位置:首页正文

数据库实验中出现的问题(回复)

来源:二三娱乐
1、查询每个学生的课程成绩最高的成绩信息(sno,cno,grade): (初始数据如下)

若执行select * from sc A where grade=(select max(grade) from sc where sno=A.sno ) order by A.sno ASC是正确的

但用select * from sc where grade in (select max(grade) from sc group by sno)的话就出现两条95001的记录,还是不明白什么原因,要是让我写语句我就会这样写。

知道为什么原因吗: 此处的原因的答案可以在实验四的第14题体现出来了. 只所以出现95001,85这条记录,是因为数据库在查找GRADE IN (92,90,85,85)成绩的时候,刚好95001当中有个学生的成绩是85分,所以它就把这个95001,85给显示出来了.

注意后者的写法是先求每个学生的最高成绩, 但这里面学号95003和95004的成绩都为85,这个没错,也出来了,关键是select * from sc where grade 这条语句的意思是求所有GRADE成绩为92,90,85,85的学生的记录, 但并没有说是求的某一个学生的最高成绩的记录. 这就是问题的根本所在.

你可以做个实验,把95001学生当中选修了2号课程的成绩85改为87, 那么得到的结论

与你想要的结论是一样的(此时不存在两个95001了).可是此时,尽管答案是符合要求,但这种写法还是错了.要深刻理解它的原理和过程. 如下图所示:

第一种求法: select * from sc A where grade=(select max(grade) from sc where sno=A.sno ) order by A.sno ASC 是一个自身连接操作.

其操作步骤是:

先求95001, 找出95001的学生的最高成绩, 再求95002, 找出95002的最高成绩. 依此类推. 请看书本102面的例35的操作过程.

14*. 查询每个学生的课程成绩最高的成绩信息(sno,cno,grade):

select * from sc A where grade=

(select max(grade) from sc where sno=A.sno ) order by A.sno ASC (此处可否用GROUP by)

select * from sc where grade in (select max(grade) from sc group by sno) (此答案有点问题: 当在数据库中同时有两个相同的最高分的,就出现判断错误.如95001里有一个最高分85, 95002里假如也有一个学生某门课成绩为85,这时问题就出现了.)

select sc.sno, maxgrade from sc , (select sno, max(grade) as maxgrade from sc group by sno) as A where sc.sno=A.sno and grade=maxgrade (这种写法是正确的, 比较特殊的一种写法)

2、查询平均成绩少于70分的学生的学号。(以下是初始数据)

用Select sno from sc group by sno having avg(grade)<70得到的数据跟分析表中应得到的一样(如下图)

执行此语句select sno from sc where not exists (Select sno from sc group by sno having avg(grade)>=70)就没有结果,当外层查询取SC表中第一个元组,而此时内层查询返回值为FALSE,那外层查询中该元组的SNO就应该放入结果表中,然后再取SC表中下一个元组啊,真的不明白为什么会这样,请老师帮忙解决下。

注意: NOT EXISTS是空集判断,当集合中不存在任何元素(为空)时,其逻辑值为true,否则为false。关键是Select sno from sc group by sno having avg(grade)>=70它得到的集合值非空, 也就是存在两个记录,如95002,95004. 因此, where not exists 这个条件没有为TRUE的情况, 那自然select sno from sc where not exists…..当然是空,没有SNO显示. 只有当where not exists 为空(也就是为TRUE)时,才会有显示的可能.

我们不妨做个实验: 将95001到95004学号的平均成绩都改成小于70分. 如:

95001 1 95001 2 95001 3 95002 2 95002 3 95003 2 95004 1 95004 2

40 45 60 66 61 69 58 70

NULL NULL NULL

再来执行: select distinct sno from sc where not exists (Select sno

from sc group by sno having avg(grade)>=70)

你会发现它把几个记录均显示出来了. 如下图所示.你放大再看.(注意distinct的用

法)

因篇幅问题不能全部显示,请点此查看更多更全内容

Top