SQL Server中是否有一个Max函数采用两个值,如.NET中的Math.Max?
时间:2020-03-06 14:38:01 来源:igfitidea点击:
我想写这样的查询:
SELECT o.OrderId, MAX(o.NegotiatedPrice, o.SuggestedPrice) FROM Order o
但这不是MAX函数的工作原理,对吗?它是一个聚合函数,因此需要一个参数,然后返回所有行的MAX。
有人知道我该怎么做吗?
解决方案
我不这么认为。前几天我想要这个。我最接近的是:
SELECT o.OrderId, CASE WHEN o.NegotiatedPrice > o.SuggestedPrice THEN o.NegotiatedPrice ELSE o.SuggestedPrice END FROM Order o
我们可以执行以下操作:
select case when o.NegotiatedPrice > o.SuggestedPrice then o.NegotiatedPrice else o.SuggestedPrice end
SELECT o.OrderID CASE WHEN o.NegotiatedPrice > o.SuggestedPrice THEN o.NegotiatedPrice ELSE o.SuggestedPrice END AS Price
我可能不会这样做,因为它的效率比前面提到的CASE构造要低,除非我们可能涵盖了两个查询的索引。无论哪种方式,它都是解决类似问题的有用技术:
SELECT OrderId, MAX(Price) as Price FROM ( SELECT o.OrderId, o.NegotiatedPrice as Price FROM Order o UNION ALL SELECT o.OrderId, o.SuggestedPrice as Price FROM Order o ) as A GROUP BY OrderId
如果我们想使用与示例类似的语法,则需要创建一个"用户定义的函数",但是我们是否可以像其他人一样使用CASE语句轻松地内联地轻松完成我们想做的事情? 。
UDF可能是这样的:
create function dbo.InlineMax(@val1 int, @val2 int) returns int as begin if @val1 > @val2 return @val1 return isnull(@val2,@val1) end
...你会这样称呼它...
SELECT o.OrderId, dbo.InlineMax(o.NegotiatedPrice, o.SuggestedPrice) FROM Order o
其他答案很好,但是如果我们不得不担心具有NULL值,则可能需要此变体:
SELECT o.OrderId, CASE WHEN ISNULL(o.NegotiatedPrice, o.SuggestedPrice) > ISNULL(o.SuggestedPrice, o.NegotiatedPrice) THEN ISNULL(o.NegotiatedPrice, o.SuggestedPrice) ELSE ISNULL(o.SuggestedPrice, o.NegotiatedPrice) END FROM Order o
我会选择kcrumley提供的解决方案
只需稍加修改即可处理NULL
create function dbo.HigherArgumentOrNull(@val1 int, @val2 int) returns int as begin if @val1 >= @val2 return @val1 if @val1 < @val2 return @val2 return NULL end
编辑
在Mark评论后修改。正如他在3个值逻辑中正确指出的那样,x> NULL或者x <NULL应该总是返回NULL。换句话说,结果未知。
糟糕,我刚刚发布了此问题的虚假信息...
答案是,没有像Oracle的Greatest这样的内置函数,但是我们可以使用UDF对2列实现类似的结果,请注意,在这里sql_variant的使用非常重要。
create table #t (a int, b int) insert #t select 1,2 union all select 3,4 union all select 5,2 -- option 1 - A case statement select case when a > b then a else b end from #t -- option 2 - A union statement select a from #t where a >= b union all select b from #t where b > a -- option 3 - A udf create function dbo.GREATEST ( @a as sql_variant, @b as sql_variant ) returns sql_variant begin declare @max sql_variant if @a is null or @b is null return null if @b > @a return @b return @a end select dbo.GREATEST(a,b) from #t
克里斯托夫
发表此答案:
create table #t (id int IDENTITY(1,1), a int, b int) insert #t select 1,2 union all select 3,4 union all select 5,2 select id, max(val) from #t unpivot (val for col in (a, b)) as unpvt group by id
可以在一行中完成:
-- the following expression calculates ==> max(@val1, @val2) SELECT 0.5 * ((@val1 + @val2) + ABS(@val1 - @val2))
编辑:如果要处理非常大的数字,则必须将值变量转换为bigint,以避免整数溢出。
DECLARE @MAX INT @MAX = (SELECT MAX(VALUE) FROM (SELECT 1 AS VALUE UNION SELECT 2 AS VALUE) AS T1)
CREATE FUNCTION [dbo].[fnMax] (@p1 INT, @p2 INT) RETURNS INT AS BEGIN DECLARE @Result INT SET @p2 = COALESCE(@p2, @p1) SELECT @Result = ( SELECT CASE WHEN @p1 > @p2 THEN @p1 ELSE @p2 END ) RETURN @Result END
对于上面关于大数的答案,我们可以在加法/减法之前进行乘法。它有点笨重,但是不需要演员。 (我不能代表速度,但我认为它仍然相当快)
SELECT 0.5 * ((@val1 + @val2) + ABS(@val1 - @val2))
更改为
SELECT @val1*0.5+@val2*0.5 + ABS(@val1*0.5 - @val2*0.5)
如果我们要避免投射,至少可以选择一种方法。
子查询可以访问外部查询中的列,因此我们可以使用此方法跨列使用汇总(例如" MAX")。 (但是,当涉及更多列时,可能会更有用)
;WITH [Order] AS ( SELECT 1 AS OrderId, 100 AS NegotiatedPrice, 110 AS SuggestedPrice UNION ALL SELECT 2 AS OrderId, 1000 AS NegotiatedPrice, 50 AS SuggestedPrice ) SELECT o.OrderId, (SELECT MAX(price)FROM (SELECT o.NegotiatedPrice AS price UNION ALL SELECT o.SuggestedPrice) d) AS MaxPrice FROM [Order] o