zoukankan      html  css  js  c++  java
  • 树状数组

    树状数组

    树状数组  重点是在树状的数组

    大家都知道二叉树吧

    叶子结点代表A数组A[1]~A[8]

     

     

     

    变形一下

     

     

     

    C[i]代表 子树的叶子结点的权值之和// 这里以求和举例

    如图可以知道

    C[1]=A[1];

    C[2]=A[1]+A[2];

    C[3]=A[3];

    C[4]=A[1]+A[2]+A[3]+A[4];

    C[5]=A[5];

    C[6]=A[5]+A[6];

    C[7]=A[7];

    C[8]=A[1]+A[2]+A[3]+A[4]+A[5]+A[6]+A[7]+A[8];

     

    将C[]数组的结点序号转化为二进制

    1=(001)      C[1]=A[1];

    2=(010)      C[2]=A[1]+A[2];

    3=(011)      C[3]=A[3];

    4=(100)      C[4]=A[1]+A[2]+A[3]+A[4];

    5=(101)      C[5]=A[5];

    6=(110)      C[6]=A[5]+A[6];

    7=(111)      C[7]=A[7];

    8=(1000)    C[8]=A[1]+A[2]+A[3]+A[4]+A[5]+A[6]+A[7]+A[8];

    1>求A数组中前i项的和

    举个例子 i=7;

    sum[7]=A[1]+A[2]+A[3]+A[4]+A[5]+A[6]+A[7] ;  

    前i项和

    C[4]=A[1]+A[2]+A[3]+A[4];  

    C[6]=A[5]+A[6];  

    C[7]=A[7];

    可以推出:   sum[7]=C[4]+C[6]+C[7];

    序号写为二进制: sum[(111)]=C[(100)]+C[(110)]+C[(111)];

    Code:

    int getsum(int i){        //求A[1 - i]的和

        int res = 0;

        while(i > 0){

            res += c[i];

            i -= lowbit(i);

        }

        return res;

    }

    2>单点更新

     

     

     

    如图:

    当更新A[1]时  需要向上更新C[1] ,C[2],C[4],C[8]

    C[1],      C[2],      C[4],       C[8]

    C[(001)],  C[(010)],   C[(100)],   C[(1000)]

     

    1                    C[1]+=A[1]

    1+lowbit(1)=2(010)     C[2]+=A[1]

    2+lowbit(2)=4(100)     C[4]+=A[1]

    4+lowbit(4)=8(1000)    C[8]+=A[1]

    Code:

    void updata(int i,int k){    //在i位置加上k

        while(i <= n){

            c[i] += k;

            i += lowbit(i);

        }

    }

    因上求缘,果上努力~~~~ 作者:每天卷学习,转载请注明原文链接:https://www.cnblogs.com/BlairGrowing/p/14096877.html

  • 相关阅读:
    vim:去掉响铃
    vim:过一个字符
    Msys2:windows下好用的unix模拟器
    vim:折叠操作
    vim:inoremap命令
    vim:关于映射和跳出括号
    vim打造简易C语言编辑器(在用2016.7.10)
    vim利用插件管理工具-管理配置文件
    拨打电话的实现
    类似于抽奖活动的小程序
  • 原文地址:https://www.cnblogs.com/BlairGrowing/p/14096877.html
Copyright © 2011-2022 走看看