zoukankan      html  css  js  c++  java
  • 线段树(Onlogn)

    #include<bits/stdc++.h>
    using namespace std;
    typedef long long ll;
    typedef struct{
    	ll l,r,w,f;
    }Node;
    Node nodes[1000000]={0,0,0};
    ll x,y,ans,mod,single;
    void lazyFlag(ll k);
    void build(ll l,ll r,ll n)
    {//建树
    	nodes[n].l=l;nodes[n].r=r;
    	if(nodes[n].l==nodes[n].r)
    	{
    		scanf("%lld",&nodes[n].w);
    		return;
    	}
    	build(l,(l+r)/2,n*2);
    	build((l+r)/2+1,r,n*2+1);
    	nodes[n].w=nodes[n*2+1].w+nodes[n*2].w;
    }
    void singleSearch(ll k)
    {//单点查询
    	if(nodes[k].l==nodes[k].r)
    	{
    		printf("%lld",nodes[k].w);
    		return;
    	}
    	if(nodes[k].f) lazyFlag(k);
    	ll mid=(nodes[k].l+nodes[k].r)/2;
    	if(single<=mid)singleSearch(k*2);
    	else singleSearch(k*2+1);
    }
    void singleChange(ll k)
    {//单点修改
    	if(nodes[k].l==nodes[k].r)
    	{
    		nodes[k].w+=mod;
    		return;
    	}
    	if(nodes[k].f) lazyFlag(k);
    	ll mid=(nodes[k].l+nodes[k].r)/2;
    	if(single<=mid)singleChange(2*k);
    	else singleChange(2*k+1);
    	//nodes[k].w+=mod;
    	nodes[k].w=nodes[2*k].w+nodes[2*k+1].w;//更改区间状态
    }
    void partSearch(ll k)
    {//区间查询
    	if(nodes[k].l>=x&&nodes[k].r<=y)
    	{
    		ans+=nodes[k].w;
    		return;
    	}
    	if(nodes[k].f) lazyFlag(k);
    	ll mid=(nodes[k].l+nodes[k].r)/2;
    	if(x<=mid)partSearch(k*2);//左
    	if(y>mid)partSearch(k*2+1);//右
    }
    void partChange(ll k)
    {//懒标记(表示区间下面的子节点等待+mod)
    	
    	if(nodes[k].l>=x&&nodes[k].r<=y)
    	{
    		nodes[k].w+=(nodes[k].r-nodes[k].l+1)*mod;
    		nodes[k].f+=mod;
    		return;
    	}
    	if(nodes[k].f) lazyFlag(k);//懒标记下传
    	ll mid=(nodes[k].l+nodes[k].r)/2;
    	if(x<=mid)partChange(k*2);
    	if(y>mid)partChange(k*2+1);
    	nodes[k].w=nodes[2*k].w+nodes[2*k+1].w;//更新区间状态
    }
    void lazyFlag(ll k)
    {
    	nodes[k*2].f+=nodes[k].f;
    	nodes[k*2+1].f+=nodes[k].f;
    	nodes[k*2].w+=nodes[k].f*(nodes[k*2].r-nodes[k*2].l+1);
    	nodes[k*2+1].w+=nodes[k].f*(nodes[k*2+1].r-nodes[k*2+1].l+1);
    	nodes[k].f=0;
    }
    int main()
    {
    	//freopen("out.out","w",stdout);
    	ll m,n;cin>>n>>m;
    	build(1,n,1);
    	//prll(n);
    	for(ll i=0;i<m;i++)
    	{
    		ll op;cin>>op;
    		if(op==1)
    		{
    			cin>>x>>y>>mod;
    			partChange(1);
    			//print(n);
    		}
    		if(op==2)
    		{
    			cin>>x>>y;
    			partSearch(1);
    			cout<<ans<<endl;
    			ans=0;
    		}
    	}
    	return 0;
    }
    
    作者:xmsi
    出处:http://www.cnblogs.com/tldr/
    本文版权归作者和博客园共有,欢迎转载,但转载时请保留此段声明。
  • 相关阅读:
    CSP-S2020总结
    题解-P6687 论如何玩转 Excel 表格
    题解-UVA12995 【Farey Sequence】
    题解-P4159 [SCOI2009] 【迷路】
    题解-SP2916【GSS5
    102. 二叉树的层序遍历
    力扣 160 相交链表 快慢指针 双指针
    3. 无重复字符的最长子串 滑动窗口
    最大连续1的个数 III
    B树和B+树
  • 原文地址:https://www.cnblogs.com/tldr/p/10877953.html
Copyright © 2011-2022 走看看