zoukankan      html  css  js  c++  java
  • 线段树,树状数组专题

    2019.7.16

    线段树,树状数组专题

    HDU 1166 敌兵布阵

    单点更新,区间求和,树状数组模板题

    #include <cstdio>
    #include <cstring>
    
    int bit[50010], n;
    
    void add(int i, int x) {
        while(i <= n) {
            bit[i] += x;
            i += i & -i;
        }
    }
    int sum(int y) {
        int s = 0;
        while(y > 0) {
            s += bit[y];
            y -= y & -y;
        }
        return s;
    }
    
    int main() {
        int t;
        scanf("%d", &t);
        for(int k = 1; k <= t; k++) {
            memset(bit, 0, sizeof(bit));
            printf("Case %d:
    ", k);
            int a;
            scanf("%d", &n);
            for(int i = 1; i <= n; i++) {
                scanf("%d", &a);
                add(i, a);
            }
            char b[10];
            int c, d;
            while(1) {
                scanf("%s", b);
                if(b[0] == 'E') break;
                else scanf("%d %d", &c, &d);
                if(b[0] == 'Q') printf("%d
    ", sum(d) - sum(c-1));
                else if(b[0] == 'A') add(c, d);
                else if(b[0] == 'S') add(c, -d);
            }
        }
        return 0;
    }
    
    

    POJ 3468 A Simple Problem with Integers

    区间修改(加或减某个数),区间求和,树状数组模板题

    #include <cstdio>
    #define lowbit(i) (i & (-i))
    typedef long long LL;
    const LL Nmax = 200100;
    LL N, Q;
    
    LL delta[Nmax];
    LL deltai[Nmax];
    LL sum[Nmax];
    
    LL Query(LL pos)
    {
    	LL temp = 0, tempp = 0, poss = pos + 1;
    	while(pos > 0)
    	{
    		temp += delta[pos];
    		tempp += deltai[pos];
    		pos -= lowbit(pos);
    	}
    	return temp * poss - tempp;
    }
    
    void Update(LL pos, LL x)
    {
        LL poss = x * pos;
    	while(pos <= N)
    	{
    		delta[pos] += x;
    		deltai[pos] += poss;
    		pos += lowbit(pos);
    	}
    }
    
    int main()
    {
    	scanf("%lld %lld", &N, &Q);
    	LL x;
    	for(LL i = 1; i <= N; ++i)
    	{
    		scanf("%lld", &x);
    		sum[i] = sum[i - 1] + x;
    	}
    	while(Q--)
    	{
    		char sign[10];
    		LL l, r, x;
    		scanf("%s", sign);
    		if(sign[0] == 'C')
    		{
    			scanf("%lld %lld %lld", &l, &r, &x);
    			Update(l, x);
    			Update(r+1, -x);
    		}
    		else
    		{
    		    scanf("%lld %lld", &l, &r);
    			LL suml = sum[l - 1] + Query(l - 1);
    			LL sumr = sum[r] + Query(r);
    			printf("%lld
    ", sumr - suml);
    		}
    	}
    
    	return 0;
    }
    

    HDU 4027 Can you answer these queries?(线段树)

    有两种操作。第一种是对某个区间的数开根,第二种操作是查询某区间的和。

    单点更新,区间求和,适当剪枝

    #include <cstdio>
    #include <cstring>
    #include <algorithm>
    using namespace std;
    typedef long long ll;
    const int N = 100000+50  ;
    ll  sum[4*N], add[4*N] ;
    void build(int l, int r, int i)
    {
        sum[i] = 0;
        if(l == r)
        {
            scanf("%lld", &sum[i]);
            return ;
        }
        int mid = (l+r)>>1;
        build(l,mid,i<<1);
        build(mid+1,r,i<<1|1);
        sum[i] = sum[i << 1] + sum[i << 1 | 1];
    }
    void update_sqrt(int i, int l, int r, int ql, int qr     ) //更新区间为qlqr,当前区间为l,r,代表当前区间和的节点为i,更新值为val,
    {
        if(l > qr || ql > r)
            return ;
        if (l>=ql&&r<=qr&&sum[i]<=r-l+1)
            return ;
        if(l==r)
        {
            sum[i]=sqrt(1.0*sum[i]);
            return ;
        }
        int mid = (l + r) >> 1;
        update_sqrt(i << 1, l, mid, ql, qr  );
        update_sqrt(i << 1 | 1, mid + 1, r, ql, qr);
        sum[i] = sum[i << 1] + sum[i << 1 | 1];
    }
     
    ll query(int i, int l, int r, int ql, int qr)     //查询区间为qlqr,当前区间为l,r,代表当前区间和的节点为i
    {
        if(l > qr || ql > r)
            return 0;
        if(l >= ql && r <= qr)
            return sum[i];
        int mid =( l + r) >> 1;
        return query(i << 1, l, mid, ql, qr)
               + query(i << 1 | 1, mid + 1, r, ql, qr);
     
    }
     
    int main()
    {
        int n,m;
        int cnt=1;
        while(scanf("%d", &n)!=EOF)
        {
            printf("Case #%d:
    ",cnt++);
            build(1,n,1);
            int m;
            cin>>m;
            int op;
            int a,b,c;
            for(  int i = 1; i <= m; i++)
            {
                scanf("%d",&op);
                if (op==1)
                {
                    scanf("%d%d",&a,&b);
                    if (a>b)swap(a,b);
                    printf("%lld
    ",query(1, 1,n,a,b) );
                }
                else if (op==0)
                {
                    scanf("%d%d",&a,&b);
                    if (a>b)swap(a,b);
                    update_sqrt(1,1,n,a,b);
                }
     
            }
            printf("
    ");
        }
        return 0;
    }
    

    HDU 1540
    https://blog.csdn.net/qq_37383726/article/details/78321600

    HDU 3974
    https://blog.csdn.net/aqa2037299560/article/details/82829151

  • 相关阅读:
    PHP 使用memcached
    linux下使用yum安装 mencached
    mysql 连接字符串 CONCAT
    linux 下 apache启动、停止、重启命令
    js中push()的用法
    linux下使用yum安装mysql
    SVN服务器多个项目的权限分组管理
    yum 安装nginx
    Linux下php安装Redis安装
    使用BarcodeLib.Barcode.ASP.NET生成条形码
  • 原文地址:https://www.cnblogs.com/fanshhh/p/11198908.html
Copyright © 2011-2022 走看看