队内训练5 线段树魔改
2021/9/16 6:06:23
本文主要是介绍队内训练5 线段树魔改,对大家解决编程问题具有一定的参考价值,需要的程序猿们随着小编来一起学习吧!
队内训练5 线段树魔改
题目链接link.
真就一眼思路然后调了一天。
题意:一个数组,两种操作,第一种区间加v,第二种求区间内 ∑ i = l r ∑ j = i + 1 r a i ∗ a j \sum_{i=l}^r\sum_{j=i+1}^r{a_i*a_j} ∑i=lr∑j=i+1rai∗aj.
这个题,就,还行,线段树进阶吧属于是,但是思路还挺好出的。首先考虑维护两个值,一个sum一个value,两个值一起更新。当两个线段合并的时候,我们可以轻易推出:
v a l u e = v a l u e 1 + v a l u e 2 + s u m 1 ∗ s u m 2 value=value1+value2+sum1*sum2 value=value1+value2+sum1∗sum2.
当然啦这只是合并的时候的式子,我们还需要考虑懒标记下放更新的时候怎么更新。
易知,当一个区间同时加上w时,更新的方式应该是:
v a l u e + = ( w 2 ∗ l e n ∗ ( l e n − 1 ) / 2 + w ∗ ( l e n − 1 ) ∗ s u m ) value+=(w^2*len*(len-1)/2+w*(len-1)*sum) value+=(w2∗len∗(len−1)/2+w∗(len−1)∗sum).
然后改一改线段树就好啦。
说一下wa一天的原因吧。
第一个wa的点是,我的 lazy_tag 初始化的值是-1,后面update的时候,直接+=w,显然很傻逼的错误,不知道怎么的误打误撞过了样例,可能对线段树过于自信也没有再手造数据疯狂改取模和全局int128,最后还是池池找出来错哪了。呜呜。
第二个wa的点,就是,可以说是奇闻共赏了:
笑拉了家人们。看看我的return 0,最离谱的是,我对拍的时候,为了方便自己找错误,直接设置了t=1,对拍跑了一年跑不出差异。笑死。
AC代码:
#include<iostream> #include<cstdio> #define int long long using namespace std; const int mod=1e9+7; int n; struct qaq { int l,r; int value=0; int sum=0; int lazy_tag=0; }tree[400100]; int a[100100]; void build(int x,int l,int r) { tree[x].l=l; tree[x].r=r; tree[x].lazy_tag=0; if(l==r) { tree[x].value=0; tree[x].sum=a[l]; return ; } int mid=(l+r)/2; build(x*2,l,mid); build(x*2+1,mid+1,r); tree[x].sum=tree[x*2].sum+tree[x*2+1].sum; tree[x].value=tree[x*2].value+tree[x*2+1].value+tree[x*2].sum*tree[x*2+1].sum; tree[x].sum%=mod; tree[x].value%=mod; } void update(int x,int w) { int len=tree[x].r-tree[x].l+1; tree[x].value += ( (((w*w)%mod)*((len*(len-1)/2)%mod)+(((w*(len-1))%mod)*tree[x].sum)%mod)%mod ); tree[x].sum += len*w;tree[x].sum%=mod; tree[x].lazy_tag+=w;tree[x].lazy_tag%=mod; return ; } void push_down(int x) { update(x*2,tree[x].lazy_tag); update(x*2+1,tree[x].lazy_tag); tree[x].lazy_tag=0; } void push_up(int x) { tree[x].sum=(tree[x*2].sum+tree[x*2+1].sum)%mod; tree[x].value=((tree[x*2].value+tree[x*2+1].value)%mod+((tree[x*2].sum*tree[x*2+1].sum)%mod)%mod)%mod; } void change(int x,int l,int r,int w) { if(tree[x].l==l&&tree[x].r==r) { update(x,w); return ; } if(tree[x].lazy_tag!=-1) push_down(x); int mid=(tree[x].l+tree[x].r)/2; if(r<=mid) change(x*2,l,r,w); else if(l>=mid+1) change(x*2+1,l,r,w); else { change(x*2,l,mid,w); change(x*2+1,mid+1,r,w); } push_up(x); } pair<int,int> query(int x,int l,int r) { if(tree[x].l==l&&tree[x].r==r) { return make_pair(tree[x].value,tree[x].sum); } if(tree[x].lazy_tag!=0) push_down(x); int mid=(tree[x].l+tree[x].r)/2; if(r<=mid) { return query(x*2,l,r); } else if(l>=mid+1) { return query(x*2+1,l,r); } else { pair<int,int> oxy=query(x*2,l,mid); pair<int,int> orz=query(x*2+1,mid+1,r); return make_pair ( (oxy.first+orz.first+(oxy.second*orz.second)%mod)%mod,(oxy.second+orz.second)%mod ); } } signed main( ) { int t; scanf("%lld",&t); while(t--) { int q; scanf("%lld%lld",&n,&q); for(int i=1;i<=n;i++) scanf("%lld",&a[i]); build(1,1,n); while(q--) { int op,l,r; scanf("%lld%lld%lld",&op,&l,&r); if(op==1)//区间修改 { int v; scanf("%lld",&v); change(1,l,r,v); } else if(op==2)//区间查询 { int ans=query(1,l,r).first; ans=(ans%mod+mod)%mod; printf("%lld\n",ans); } } } return 0; }
这篇关于队内训练5 线段树魔改的文章就介绍到这儿,希望我们推荐的文章对大家有所帮助,也希望大家多多支持为之网!
- 2025-01-11cursor试用出现:Too many free trial accounts used on this machine 的解决方法
- 2025-01-11百万架构师第十四课:源码分析:Spring 源码分析:深入分析IOC那些鲜为人知的细节|JavaGuide
- 2025-01-11不得不了解的高效AI办公工具API
- 2025-01-102025 蛇年,J 人直播带货内容审核团队必备的办公软件有哪 6 款?
- 2025-01-10高效运营背后的支柱:文档管理优化指南
- 2025-01-10年末压力山大?试试优化你的文档管理
- 2025-01-10跨部门协作中的进度追踪重要性解析
- 2025-01-10总结 JavaScript 中的变体函数调用方式
- 2025-01-10HR团队如何通过数据驱动提升管理效率?6个策略
- 2025-01-10WBS实战指南:如何一步步构建高效项目管理框架?