原文:SQL点滴23—T-SQL中的除法

  

在T-SQL中没有除法运算,但是在T-SQL中可以实现类似除法的操作Divide。一般除法操作的结果一个列来自于被除关系表,剩下的来自除关系表。这里举一个例子来说明。假设如下有三个表:客户Customers,销售人员Employees,订单Orders,查询返回一些客户,要求这些客户和所有美国雇员都至少有一次交易记录。来看下面一个语句:

select custid from Sales.Customers as C
where not exists
(select * from HR.Employees as E
where country='USA'
and not exists
(select * from Sales.Orders as O
where O.custid = C.custid
and O.empid = E.empid))

  

语句返回23行,标明有23名客户,这些客户至少和每个美国雇员有一次交易记录。现在我们如果修改一下条件,问题要求还是一样的,我们把销售人员的国家修改成以色列Israel,看看以色列的销售人员是否能像美国雇员一样的强悍。

  

select custid from Sales.Customers as C
where not exists
(select * from HR.Employees as E
where country='Israel'
and not exists
(select * from Sales.Orders as O
where O.custid = C.custid
and O.empid = E.empid))

  

修改国家条件,这次我们查询得到的结果是91条记录,我们看看Customers这个表总共也就91条记录,很明显这个结果不对。我们使用语句来看看select * from Sales.Customers where country like '%Israel%',查询得到0条记录,就是说根本就没有以色列的雇员。因为没有来自以色列的雇员,所有雇员和该客户拥有至少一项交易记录这个条件对每个雇员都满足,这个是代数里面的空真现象。换句话说每个客户都和这个不存在的以色列雇员至少有一项交易记录。这个很像除法运算的一个规则:除数是0,商就是无限大

  
写上面的语句的时候,我们没有考虑如果Employee表中没有如果没有以色列的雇员怎么办。如果我们在问题中加上确实存在来自以色列的雇员就可以避免这个错误,只需要在条件中限定至少有一个以色列雇员存在于表Employee中就可以了。这个就像除法中的非0限定:除数不为0。语句如下:

  

select custid from Sales.Customers as C
where not exists
(select * from HR.Employees as E
where country='Israel'
and not exists
(select * from Sales.Orders as O
where O.custid = C.custid
and O.empid = E.empid))
and exists (select * from HR.Employees as E where country='Israel')

现在查询得到0条结果,这才是我们想要的。

  
在这个除法操作中有三个关系,a Divide by b Per c,a是被除数,b是除数,c是中介关系。假设a有属性A,b有属性B。在上面的语句中被除关系是Customers,除数关系是满足一定关系的Employee,中介关系是Orders。这里为了避免除数是0 的问题,使用第四个临时关系(select * from HR.Employees as E where country='Israel')。也可以使用另外一种方法来限定至少有一名以色列的销售人员和所有顾客至少有一次交易记录。如下:
a,找到美国雇员的id:select empid from HR.Employees where country='USA',这个语句找到的是(1,2,3,4,8)。
b,找到美国雇员的数量:select COUNT(*) from HR.Employees where country='USA',很明显这个找到的结果是5。
c,根据雇员id查找交易记录表中的客户id,并按客户id分组,在分组中查找不重复的empid数目等于5的。

  

select custid
from Sales.Orders
where empid in (1,2,3,4,8) group by custid having count(distinct empid)=5

  

我们把上面两个查询还原上去,由于没有以色列的雇员,还是使用美国雇员:

select custid
from Sales.Orders
where empid in
(select empid from HR.Employees where country = N'USA')
group by custid
having count(distinct empid) = (select count(*) from HR.Employees where country = N'USA')

查询得到的结果是23条记录,符合我们的要求。

 

SQL点滴23—T-SQL中的除法的更多相关文章

  1. SQL点滴15—在SQL Server 2008中调用C#程序

    原文:SQL点滴15-在SQL Server 2008中调用C#程序 T-SQL的在执行普通的查询的时候是很高效的,但是在执行循环,判断这样的语句的时候效率就不那么的高了.这时可以借助CLR了,我们可 ...

  2. SQL点滴2—重温sql语句中的join操作

    原文:SQL点滴2-重温sql语句中的join操作 1.join语句 Sql join语句用来合并两个或多个表中的记录.ANSI标准SQL语句中有四种JOIN:INNER,OUTER,LEFTER,R ...

  3. SQL点滴7—使用SQL Server的attach功能出现错误及解决方法

    原文:SQL点滴7-使用SQL Server的attach功能出现错误及解决方法 今天用SQL Server 2008的attach功能附加一个数据库,出了点问题,提示的错误是: Unable to ...

  4. Azure SQL Database (23) Azure SQL Database Dynamic Data Masking动态数据掩码

    <Windows Azure Platform 系列文章目录> 我们在使用关系型数据的时候,有时候希望: - 管理员admin,可以查看到所有的数据 - 普通用户,某些敏感字段,比如信用卡 ...

  5. SQL点滴17—使用数据库引擎存储过程,系统视图查询,DBA,BI开发人员必备基础知识

    原文:SQL点滴17-使用数据库引擎存储过程,系统视图查询,DBA,BI开发人员必备基础知识 在开发过程中会遇到需要弄清楚这个数据库什么时候建的,这个数据库中有多少表,这个存储过程长的什么样子等等信息 ...

  6. sql点滴42—mysql中的时间转换

    原文:sql点滴42-mysql中的时间转换 UNIX时间戳转换为日期用函数: FROM_UNIXTIME() select FROM_UNIXTIME(1156219870); 日期转换为UNIX时 ...

  7. sql点滴42—mysql中的数据结构

    原文:sql点滴42-mysql中的数据结构 MySQL 的数值数据类型可以大致划分为两个类别,一个是整数,另一个是浮点数或小数.许多不同的子类型对这些类别中的每一个都是可用的,每个子类型支持不同大小 ...

  8. SQL点滴19—T-SQL中的透视和逆透视

    原文:SQL点滴19-T-SQL中的透视和逆透视 透视 今天抽一点时间来看看透视和逆透视语句,简单的说就是行列转换.假设一个销售表中存放着产品号,产品折扣,产品价格三个列,每一种产品号可能有多种折扣, ...

  9. SQL点滴9—SQL Server中的事务处理以及SSIS中的内建事务

    原文:SQL点滴9-SQL Server中的事务处理以及SSIS中的内建事务 我们可以把SSIS中的整个package包含在一个事务中,但是如果在package的执行过程中有一个表需要锁定应该怎么处理 ...

随机推荐

  1. C语言,如何检查文件是否存在和权限的信息

    按功能access,头文件io.h(linux通过使用unistd.h    int   access(const   char   *filename,   int   amode); amode參 ...

  2. JAVA多线程两个实用的辅助类(CountDownLatch和AtomicBoolean)

    AtomicBoolean它允许一个线程等待一个线程完成任务,然后运行: A boolean value that may be updated atomically. See the java.ut ...

  3. 超赞的CSS3进度条 可以随进度显示不同颜色

    原文:超赞的CSS3进度条 可以随进度显示不同颜色 现在的WEB已经不是以前的WEB了,传输更大的数据量,有着更加复杂的计算,这就需要利用进度条来提高用户体验,必要时可以让用户耐心等待,不至于因操作卡 ...

  4. 炫酷的CSS3发光搜索表单,附演示和源码

    原文:炫酷的CSS3发光搜索表单,附演示和源码 昨天在国外的论坛上逛的时候,看到一篇帖子是求助如何利用CSS3实现发光效果的,网友回复中有一个推荐了一款CSS3发光搜索表单,利用CSS3的动画属性,设 ...

  5. tc-SRM-626-DIV1-250

    枚举alice投掷骰子得到的结果的每一种情况极其数量. 枚举bob投掷骰子得到的结果的每一种情况极其数量. 然后枚举alice投掷骰子得到的结果的数量和bob投掷骰子比alice低的数量. 然后计算结 ...

  6. oracle从备份归档日志的方法集中回收

    oracle从备份集中抓出归档日志方法 在大连医院遇到这个问题,数据库为归档状态,但归档完成后rman通过crontab自己主动备走归档日志并删除存在系统上的归档日志文件.在RealSync程序停止一 ...

  7. JavaScript 实现Map效果

    var map = {}; // 类似:Map map = new HashMap(); map[key] = value; // 类似:map.put(key, value); var value ...

  8. shell 监控局域网的主机是否up(转)

    #!/bin/bash for ((i=30;i<60;i++)) ;do ping -c 3 172.31.0.$i>/dev/null #ping -c 172.31.0.30 ~17 ...

  9. ABP领域层——工作单元(Unit Of work)

    ABP领域层——工作单元(Unit Of work) 点这里进入ABP系列文章总目录 基于DDD的现代ASP.NET开发框架--ABP系列之12.ABP领域层——工作单元(Unit Of work) ...

  10. DevExpress XtraReports 入门六 控件以程序方式创建一个 交叉表 报表

    原文:DevExpress XtraReports 入门六 控件以程序方式创建一个 交叉表 报表 本文只是为了帮助初次接触或是需要DevExpress XtraReports报表的人群使用的,为了帮助 ...