行为不当(Misbehaving)的子查询
2008-11-13 22:36
453 查看
子查询有一个不知道是不是bug的设计问题.曾经出过一个错误,幸好在投入使用前发现问题,避免错误.
要在数据库中返回数据:没有销售过商品给会员'EDWIN'的销售员工.SQL很简单,马上写道:
SELECT StaffID,StaffName
FROM dbo.MsStaff
WHERE StaffID NOT IN
(SELECT StaffID FROM dbo.TxSales
WHERE VipCode='EDWIN')
奇怪的是,它返回一个空集,从业务上看,很明显是错误的,因为不可能全国所有员工都卖过商品给该会员.检查发现是因为TxSales表中的销售员工字段不叫StaffID,而是SalesMan.非常奇怪为什么SQL不报"无效列名"错误,而当我单独运行子查询部分时,是正常报错的.
翻阅资料,得知:在外部查询的上下文中,该子查询是有效的!名称解释进行从内部嵌套级别向外进行.查询处理器先在当前级别引用的TxSales表中寻找StaffID列,未找到这个列名称后,它将在MsStaff表(外部级别)中查找并找到该列.子查询变成了相关子查询:
SELECT StaffID,StaffName
FROM dbo.MsStaff As M
WHERE StaffID NOT IN
(SELECT M.StaffID FROM dbo.TxSales As S
WHERE S.VipCode='EDWIN')
逻辑上,该查询没有意义,但它在技术上是可行的.这样的SQL结果只有两种情况:
1)销售单中没有EDWIN这个会员,那么查询返回所有的销售员工
2)如上例,返回空集.
记录一下,以后写SQL要看清楚结果是否符合现实情况.数据是否在合理范围,因为通常错误的SQL返回的结果和业务数据差距会比较大,通过简单对比能够检查出部分错误.
要在数据库中返回数据:没有销售过商品给会员'EDWIN'的销售员工.SQL很简单,马上写道:
SELECT StaffID,StaffName
FROM dbo.MsStaff
WHERE StaffID NOT IN
(SELECT StaffID FROM dbo.TxSales
WHERE VipCode='EDWIN')
奇怪的是,它返回一个空集,从业务上看,很明显是错误的,因为不可能全国所有员工都卖过商品给该会员.检查发现是因为TxSales表中的销售员工字段不叫StaffID,而是SalesMan.非常奇怪为什么SQL不报"无效列名"错误,而当我单独运行子查询部分时,是正常报错的.
翻阅资料,得知:在外部查询的上下文中,该子查询是有效的!名称解释进行从内部嵌套级别向外进行.查询处理器先在当前级别引用的TxSales表中寻找StaffID列,未找到这个列名称后,它将在MsStaff表(外部级别)中查找并找到该列.子查询变成了相关子查询:
SELECT StaffID,StaffName
FROM dbo.MsStaff As M
WHERE StaffID NOT IN
(SELECT M.StaffID FROM dbo.TxSales As S
WHERE S.VipCode='EDWIN')
逻辑上,该查询没有意义,但它在技术上是可行的.这样的SQL结果只有两种情况:
1)销售单中没有EDWIN这个会员,那么查询返回所有的销售员工
2)如上例,返回空集.
记录一下,以后写SQL要看清楚结果是否符合现实情况.数据是否在合理范围,因为通常错误的SQL返回的结果和业务数据差距会比较大,通过简单对比能够检查出部分错误.
相关文章推荐
- ReportStudio入门教程(九十二) - 追溯行为配置不当引发的错误
- 经深圳市公安局调查,认定猥亵罪不成立,仅为酒后行为不当
- Xfire向雅虎提起反诉讼 要求就不当行为赔偿
- 9.3 行为不当的机器人
- 为什么某些Win32技术在Windows NT服务中行为不当?
- SQL中行为不当的子查询,IN,exists
- 软件工程师的十个“不职业”行为
- 对象行为模式——中介者模式(Mediator)
- 十个糟糕的程序员的行为
- 别拿腰痛不当肾病
- AutoResetEvent 的诡异行为
- Android启动模式对activity行为的影响
- 基于Linux的行为查杀的反病毒软件的一二思路
- 成功项目经理的三种领导力行为
- Context 使用不当造成内存泄露
- 高效程序员的40个好习惯和行为方式
- 关于radio的onchange事件在IE、FF中行为不同的问题
- CSS中的行为——expression
- Spring7种事务传播行为类型