前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >一种递推组合数前缀和的Trick

一种递推组合数前缀和的Trick

作者头像
attack
发布2018-10-08 11:44:26
9820
发布2018-10-08 11:44:26
举报

记录一下一种推组合数前缀和的方法

Trick

设\(\sum_{i = 0}^m C_n^i = S(n, m)\)

\(S\)是可以递推的

  • \(S(n, m + 1) = S(n, m) + C_{n}^{m + 1}\)

就是加上最末尾的一项

  • \(S(n + 1, m) = 2S(n, m) - C_n^m\)

\(S(n, m)\)可以看做是杨辉三角上的一行,而\(S(n+1, m)\)是他的下一行

考虑组合数的递推公式,除了\(C[n][m]\)这一项之外都会被计算两次、

另外如果有多组询问的话可以用莫队实现

代码语言:javascript
复制
#include<bits/stdc++.h>
using namespace std;
int N, M, Lim, C[1001][1001], S[1001][1001];
int Make1() {
    for(int i = 0; i <= Lim; i++) {
        C[i][0] = C[i][i] = 1;
        for(int j = 1; j < i; j++)
            C[i][j] = C[i - 1][j - 1] + C[i - 1][j];        
    }
    return C[N][M];
}
void Make2() {
    /*for(int i = 0; i <= Lim; i++) {
        S[i][0] = 1;
        for(int j = 1; j <= i; j++)
            S[i][j] = S[i][j - 1] + C[i][j];        
    }*/
    for(int i = 0, base = 1; i <= Lim; i++, base <<= 1) {
        S[i][i] = base;
        for(int j = i + 1; j <= Lim; j++)
            S[j][i] = 2 * S[j - 1][i] - C[j - 1][i];        
    }
}
void print(int (*a)[1001]) {
    for(int i = 0; i <= Lim; i++, puts("")) {
        for(int j = 0; j <= i; j++)
            printf("%d ", S[i][j]);     
    }
}
main() {
    Lim = 10;
    //cin >> N >> M;
    Make1();
    Make2();
    print(S);
    
}
本文参与 腾讯云自媒体分享计划,分享自作者个人站点/博客。
原始发表:2018-10-07 ,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 作者个人站点/博客 前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体分享计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
目录
  • Trick
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档