干杯,
我遇到了一个挑战,我曾经多次遇到但从未能找到有效的解决方案.想象一下,我有一个大表,其中包含有关例如银行账户及其从借方到贷方的可能循环转移:
- AccountId DebitCredit AsOfDate
- --------- ----------- ----------
- aaa d 2018-11-01
- aaa d 2018-11-02
- aaa c 2018-11-03
- aaa c 2018-11-04
- aaa c 2018-11-05
- bbb d 2018-11-02
- ccc c 2018-11-01
- ccc d 2018-11-02
- ccc d 2018-11-03
- ccc c 2018-11-04
- ccc d 2018-11-05
- ccc c 2018-11-06
在上面的示例中,我想将子分区号分配给AccountId和DebitCredit的组合,其中每次DebitCredit移位时分区号都会递增.换句话说,在上面的示例中,我希望得到以下结果:
- AccountId DebitCredit AsOfDate PartNo
- --------- ----------- ---------- ------
- aaa d 2018-11-01 1
- aaa d 2018-11-02 1
- aaa c 2018-11-03 2
- aaa c 2018-11-04 2
- aaa c 2018-11-05 2
- bbb d 2018-11-02 1
- ccc c 2018-11-01 1
- ccc d 2018-11-02 2
- ccc d 2018-11-03 2
- ccc c 2018-11-04 3
- ccc d 2018-11-05 4
- ccc c 2018-11-06 5
我无法真正弄清楚如何快速有效地做到这一点.该操作必须每天在具有数百万行的表上完成.
在这个具体的例子中,我们保证所有帐户都有连续的行.但是,客户当然可以在当月15日开立账户和/或在第26天关闭账户.
在MSsql 2016服务器上具体要解决这个挑战,但是可以在2012(甚至可能是2008r2)上运行的解决方案会很好.
您可以想象,无法确定是否只有借记卡或信用额度,或者该帐户是否每天都会循环使用.
我真的希望有人可以帮助我:)
谢谢
解决方法
如果你有sql server 2012,你可以使用lag()和窗口求和来得到这个:
- select *,sum(PartNoAdd) over (partition by AccountId order by AsOfDate asc) as PartNo_calc
- from
- (
- select *,case when DebitCredit=lag(DebitCredit,1) over (partition by AccountId order by AsOfDate asc) then 0 else 1 end as PartNoAdd
- from t
- )t2
- order by AccountId asc,AsOfDate asc
在内部查询中,PartNoAdd检查此帐户的先前DebitCard是否相同.如果是,则返回0(我们不应该添加任何内容),否则返回1.
然后外部查询将此帐户的所有PartNoAdd相加.