隨筆 - 312, 文章 - 14, 評(píng)論 - 1393, 引用 - 0
          數(shù)據(jù)加載中……

          妙用SQL Server聚合函數(shù)和子查詢迭代求和

          本文為原創(chuàng),如需轉(zhuǎn)載,請(qǐng)注明作者和出處,謝謝!

          先看看下面的表和其中的數(shù)據(jù):

          t_product


                      圖1

          該表有兩個(gè)字段:xhprice 其中xh是主索引字段,現(xiàn)在要得到如下的查詢結(jié)果:

              圖2

          從上面的查詢結(jié)果可以看出,totalprice字段值的規(guī)則是從第1條記錄到當(dāng)前記錄的price之和。如第3條記錄的totalprice字段的值是10 + 25 + 36 = 71

          現(xiàn)在要通過t_product表中的數(shù)據(jù)生成圖2所示的查詢結(jié)果。可能會(huì)有很多讀者想到使用循環(huán)和游標(biāo),不過這種方式效率并不高,尤其在記錄非常多的情況。

          從圖2的查詢結(jié)果分析可知,這個(gè)結(jié)果仍然是求和的操作,只是并不是對(duì)所有的記錄求和,也不是分組求和,而是使用迭代的方式進(jìn)行求和,求和的公式如下:

          當(dāng)前記錄的totalprice = 當(dāng)前記錄的price + 上一條記錄的totalprice

          上一條記錄的totalprice值也可看成是當(dāng)前記錄以前所有記錄的price值之和。因此,可以對(duì)每一條記錄進(jìn)行求和(使用sum函數(shù)),不過要求出當(dāng)前記錄及以前的記錄的price之和,如下面的SQL語句:

          select a.xh, a.price,
          (
          select sum(price) from t_product b where b.xh <= a.xh) as totalprice 
          from t_product a

          從上面的SQL語句可以看出,使用了一個(gè)子查詢來求totalprice字段的值,基本原理就是根據(jù)當(dāng)前記錄的xh值(a.xh)來計(jì)算從當(dāng)前記錄往前所有記錄的price值之和,b.xh表示子查詢當(dāng)前的xh值,在子查詢中,a.xh相當(dāng)于常量。上面的SQL語句的查詢結(jié)果和圖2完全一樣。如果我們的需求是不包含當(dāng)前記錄的price值,也就是說,計(jì)算totalprice字段的公式如下:

          當(dāng)前記錄的totalprice = 上一條當(dāng)前記錄的price + 上一條記錄的totalprice

          第一條記錄的totalprice值就是當(dāng)前記錄的price值,查詢t_product表的結(jié)果如圖3所示。


          3

          要查詢出上述的記錄也很容易,只需要將<=改成<即可,SQL語句如下:


          select a.xh, a.price,
          (
          select sum(price) from t_product b where b.xh < a.xh) as totalprice 
          from t_product a

          但上面的SQL查詢出來的記錄的第一條的totalprice字段值為null,如圖4所示。


                圖4

          為了將這個(gè)null換成10,可以使用case語句,SQL語句如下:


          select xh, price, 
          (
          case  when totalprice is null then price else totalprice end ) as totalprice
          from
          (
          select a.xh, (select  sum(price) from t_product b where b.xh < a.xh)  as totalprice , a.price
          from t_product a)  x

          在上面的SQL語句共有三層select查詢,最里面一層如下:

          select  sum(price) from t_product b where b.xh < a.xh)

          中間一層的子查詢?nèi)缦拢?/span>

          select a.xh, (select  sum(price) from t_product b where b.xh < a.xh)  as totalprice , a.price
          from t_product a

          最外面一層當(dāng)然就是整個(gè)select語句了。

          在執(zhí)行上面的SQL后,將會(huì)得到和圖3一樣的查詢結(jié)果了。

          如果讀者不喜歡寫太長的SQL,可以將部分內(nèi)容寫到函數(shù)里,代碼如下:

          create function mysum(@xh int@price intreturns int
          begin
            
          return (select 
                    (
          case when totalprice is null then @price  else totalprice endas totalprice 
                   
          from ( select  sum(price) as totalprice from t_product where xh < @xh) x)
          end

          可使用下面的SQL語句來使用這個(gè)函數(shù):

          select xh, price, dbo.mysum(xh, price)  as totalprice
          from t_product

          在執(zhí)行上面的SQL后,將得出如圖3所示的查詢結(jié)果。

          建立t_product表的SQL語句(SQL Server 2005)如下:

          SET ANSI_NULLS ON
          GO
          SET QUOTED_IDENTIFIER ON
          GO
          IF NOT EXISTS (SELECT * FROM sys.objects WHERE object_id = OBJECT_ID(N'[dbo].[t_product]'AND type in (N'U'))
          BEGIN
          CREATE TABLE [dbo].[t_product](
              
          [xh] [int] NOT NULL,
              
          [price] [int] NOT NULL,
           
          CONSTRAINT [PK_t_product] PRIMARY KEY CLUSTERED 
          (
              
          [xh] ASC
          )
          WITH (IGNORE_DUP_KEY = OFFON [PRIMARY]
          ON [PRIMARY]
          END





          Android開發(fā)完全講義(第2版)(本書版權(quán)已輸出到臺(tái)灣)

          http://product.dangdang.com/product.aspx?product_id=22741502



          Android高薪之路:Android程序員面試寶典 http://book.360buy.com/10970314.html


          新浪微博:http://t.sina.com.cn/androidguy   昵稱:李寧_Lining

          posted on 2008-09-02 12:53 銀河使者 閱讀(2306) 評(píng)論(3)  編輯  收藏 所屬分類: SQL Serverdatabases 原創(chuàng)

          評(píng)論

          # re: 妙用SQL Server聚合函數(shù)和子查詢迭代求和  回復(fù)  更多評(píng)論   

          博主那名字總讓我會(huì)想起:李銀河
          2008-09-02 14:16 | 隔葉黃鶯

          # re: 妙用SQL Server聚合函數(shù)和子查詢迭代求和  回復(fù)  更多評(píng)論   

          李銀河,很耳熟,是誰?
          2008-09-02 14:39 | 銀河使者

          # re: 妙用SQL Server聚合函數(shù)和子查詢迭代求和  回復(fù)  更多評(píng)論   

          好,喜歡~贊一個(gè)
          2008-09-02 20:19 | 試客網(wǎng)
          主站蜘蛛池模板: 阿克陶县| 吉首市| 禄劝| 英德市| 措勤县| 柘荣县| 定陶县| 鸡西市| 曲松县| 开鲁县| 麟游县| 宁武县| 平南县| 阜城县| 桦南县| 南华县| 宁化县| 北流市| 来凤县| 大方县| 庆阳市| 阿拉善盟| 苍山县| 长岛县| 开化县| 乐业县| 萍乡市| 昂仁县| 隆回县| 迁西县| 新巴尔虎右旗| 澄迈县| 上犹县| 凤台县| 山西省| 张家川| 北海市| 航空| 屏山县| 依安县| 湟源县|