洛谷传送门

CF266E传送门


分析

首先区间修改区间查询首选线段树

要找突破口,\((i-l+1)^k\)中\(i\)不是定值,

显然得拆开,而且\(k\)很小,根据二项式定理,

\[\sum_{i=l}^ra_i(i-l+1)^k=\sum_{i=l}^ra_i\sum_{j=0}^kC(k,j)\times i^j\times (l-1)^{k-j}\times (-1)^{k-j}
\]
\[=\sum_{j=0}^kC(k,j)\times (\sum_{i=l}^ra_i\times i^j)\times (l-1)^{k-j}\times (-1)^{k-j}
\]

那么线段树叶子节点维护\(a_i\times i^j\),

对于区间修改在预处理时做前缀和,直接赋值

显然正整数幂和组合数都能预处理,

所以时间复杂度\(O((n+m)klogn)\)

卡常:

  1. 不用快速幂,预处理的时候通过\(i^{j-1}\)推出\(i^j\)
  2. 注意0^0=1,虽然在初中无意义
  3. 线段树每次只递归1次,\(j=0\sim 5\)同时修改查询,查询最后用一个结构体记录答案,懒标记只需要1个
  4. 组合数可以不用阶乘,实测快30ms

代码

#include <cstdio>
#include <cctype>
#define rr register
using namespace std;
const int N=100011,mod=1000000007;
int si[6][N],a[N],n,m,c[6][6];
inline signed iut(){
rr int ans=0; rr char c=getchar();
while (!isdigit(c)) c=getchar();
while (isdigit(c)) ans=(ans<<3)+(ans<<1)+(c^48),c=getchar();
return ans;
}
inline void print(int ans){
if (ans>9) print(ans/10);
putchar(ans%10+48);
}
inline signed mo(int x,int y){return x+y>=mod?x+y-mod:x+y;}
inline void Pro(int n){
c[0][0]=1;
for (rr int i=1;i<6;++i){
c[i][0]=c[i][i]=1;
for (rr int j=1;j<i;++j)
c[i][j]=c[i-1][j-1]+c[i-1][j];
}
for (rr int i=0;i<=n;++i) si[0][i]=1;
for (rr int j=1;j<6;++j)
for (rr int i=1;i<=n;++i) si[j][i]=1ll*si[j-1][i]*i%mod;
for (rr int j=0;j<6;++j)
for (rr int i=1;i<=n;++i) si[j][i]=mo(si[j][i],si[j][i-1]);
}
inline signed Get(int k,int l,int r){return mo(si[k][r],mod-si[k][l-1]);}
struct Array_Six{int Q[6];};
struct Segment_Tree{
int w[N<<2][6],lazy[N<<2];
inline void pup(int k){
for (rr int j=0;j<6;++j)
w[k][j]=mo(w[k<<1][j],w[k<<1|1][j]);
}
inline void build(int k,int l,int r){
lazy[k]=-1;
if (l==r){
for (rr int j=0;j<6;++j)
w[k][j]=1ll*a[l]*Get(j,l,l)%mod;
return;
}
rr int mid=(l+r)>>1;
build(k<<1,l,mid);
build(k<<1|1,mid+1,r);
pup(k);
}
inline void pdown(int k,int l,int mid,int r){
if (~lazy[k]){
lazy[k<<1]=lazy[k<<1|1]=lazy[k];
for (rr int j=0;j<6;++j)
w[k<<1][j]=1ll*lazy[k]*Get(j,l,mid)%mod,
w[k<<1|1][j]=1ll*lazy[k]*Get(j,mid+1,r)%mod;
lazy[k]=-1;
}
}
inline void update(int k,int l,int r,int x,int y,int z){
if (l==x&&r==y){
lazy[k]=z;
for (rr int j=0;j<6;++j)
w[k][j]=1ll*z*Get(j,l,r)%mod;
return;
}
rr int mid=(l+r)>>1; pdown(k,l,mid,r);
if (y<=mid) update(k<<1,l,mid,x,y,z);
else if (x>mid) update(k<<1|1,mid+1,r,x,y,z);
else update(k<<1,l,mid,x,mid,z),update(k<<1|1,mid+1,r,mid+1,y,z);
pup(k);
}
inline Array_Six query(int k,int l,int r,int x,int y){
rr Array_Six t1,t2;
if (l==x&&r==y){
for (rr int j=0;j<6;++j)
t1.Q[j]=w[k][j];
return t1;
}
rr int mid=(l+r)>>1; pdown(k,l,mid,r);
for (rr int j=0;j<6;++j) t1.Q[j]=t2.Q[j]=0;
if (y<=mid) t1=query(k<<1,l,mid,x,y);
else if (x>mid) t2=query(k<<1|1,mid+1,r,x,y);
else t1=query(k<<1,l,mid,x,mid),
t2=query(k<<1|1,mid+1,r,mid+1,y);
for (rr int j=0;j<6;++j) t1.Q[j]=mo(t1.Q[j],t2.Q[j]);
return t1;
}
}Tre;
signed main(){
n=iut(); m=iut(); Pro(N-11);
for (rr int i=1;i<=n;++i) a[i]=iut();
Tre.build(1,1,n);
for (;m;--m){
rr char C=getchar();
while (C!='?'&&C!='=') C=getchar();
if (C=='='){
rr int l=iut(),r=iut(),x=iut();
Tre.update(1,1,n,l,r,x);
}else{
rr int l=iut(),r=iut(),k=iut(),ans=0;
rr Array_Six t=Tre.query(1,1,n,l,r);
for (rr int j=0,W;j<=k;++j){
if (l>1) W=Get(k-j,l-1,l-1);
else W=si[k-j][l-1];
if (((k^j)&1)&&W) W=mod-W;
ans=mo(ans,1ll*W*c[k][j]%mod*t.Q[j]%mod);
}
print(ans),putchar(10);
}
}
return 0;
}

#线段树,组合计数,二项式定理#CF266E More Queries to Array的更多相关文章

  1. 线段树维护矩阵【CF718C】 Sasha and Array

    Description 有一个长为\(n\)的数列\(a_{1},a_{2}...a_{n}\),你需要对这个数列维护如下两种操作: \(1\space l \space r\space x\) 表示 ...

  2. 数据结构(线段树):CodeForces 145E Lucky Queries

    E. Lucky Queries time limit per test 3 seconds memory limit per test 256 megabytes input standard in ...

  3. 牛客 2B 树 (组合计数)

    传送门 大意: 给定n节点树, 求划分为不超过$k$个连通块的方案数. n,k<=300. 核心观察是每个连通块深度最低的点固定以后染色方案就固定了. 所以答案为$\sum\limits_{i= ...

  4. Codeforces Round #312 (Div. 2) E. A Simple Task 线段树

    E. A Simple Task 题目连接: http://www.codeforces.com/contest/558/problem/E Description This task is very ...

  5. 线段树 Interval Tree

    一.线段树 线段树既是线段也是树,并且是一棵二叉树,每个结点是一条线段,每条线段的左右儿子线段分别是该线段的左半和右半区间,递归定义之后就是一棵线段树. 例题:给定N条线段,{[2, 5], [4, ...

  6. [HDOJ5877]Weak Pair(DFS,线段树,离散化)

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=5877 题意:给一棵树和各点的权值a,求点对(u,v)个数,满足:1.u是v的祖先,2.a(u)*a(v ...

  7. HDU 5266 pog loves szh III 线段树,lca

    Pog and Szh are playing games. Firstly Pog draw a tree on the paper. Here we define 1 as the root of ...

  8. [Codeforces 266E]More Queries to Array...(线段树+二项式定理)

    [Codeforces 266E]More Queries to Array...(线段树+二项式定理) 题面 维护一个长度为\(n\)的序列\(a\),\(m\)个操作 区间赋值为\(x\) 查询\ ...

  9. 暑假集训单切赛第一场 CF 266E More Queries to Array(线段树+二项式展开式)

    比赛时,第二题就是做的这个,当时果断没仔细考虑,直接用线段树暴力求.结果易想而知,超时了. 比赛后搜了搜题解,恍然大悟. 思路:显然用线段树,但是由于每次查询都会有变,所以不可能存储题目中的式子.   ...

  10. Codeforces Round #312 (Div. 2) E. A Simple Task 线段树+计数排序

    题目链接: http://codeforces.com/problemset/problem/558/E E. A Simple Task time limit per test5 secondsme ...

随机推荐

  1. 一个有意思的问题:Kafka的消费Offset会溢出吗

    最近在项目上接入公司APP产品的用户点击日志数据时,发现消费者组的Offset值非常大,才一天的时间,已提交的Offset值就有千亿级别了. 于是不禁想了一个问题:假设一个Topic就只有一个Part ...

  2. SpringCloud Sentinel使用

    1. 简介 Sentinel 是面向分布式服务架构的流量控制组件,主要以流量为切入点,从限流.流量整形.熔断降级.系统负载保护.热点防护等多个维度来帮助开发者保障微服务的稳定性.替换原先Hystrix ...

  3. Sermant运行流程学习笔记,速来抄作业

    本文分享自华为云社区<Sermant 的整体流程学习梳理>,作者:用友汽车信息科技(上海)有限公司 刘亚洲 Java研发工程师. 一.sermant架构 Sermant整体架构包括Serm ...

  4. 从零开始搭建Springboot开发环境(Java8+Git+Maven+MySQL+Idea)之一步到位

    说明 所谓万事开头难,对于初学Java和Springboot框架的小伙伴往往会花不少时间在开发环境搭建上面.究其原因其实还是不熟悉,作为在IT界摸爬滚打数年的老司机,对于各种开发环境搭建已经了然于胸, ...

  5. STM32F103xC,xD,xE引脚定义

    STM32F103xC,xD,xE引脚定义 由于在使用STM32系列芯片过程中发现互联网没有整理好的引脚定义,因此自己整理一份,方便以后查阅. GPIOA Pin 重新上电时的功能 默认功能 重映射 ...

  6. vue配置scss全局样式

    安装插件 npm install sass --save-dev 在src文件夹下创建styles文件夹,并创建以下文件 index.scss: scss的入口文件 // 引入清除默认样式 @impo ...

  7. day01-项目介绍与环境搭建

    项目介绍与环境搭建 1.项目学习前置知识 Java基础知识 javaweb MySQL SpringBoot SSM(Spring,SpringMVC,MyBatis) Maven 2.学习收获 了解 ...

  8. Git进阶命令-reset

    之前有关Git,写过一片文章: Git五个常见问题及解决方法 一.reset命令使用场景 有时候我们提交了一些错误的或者不完善的代码,需要回退到之前的某个稳定的版本,面对这种情况有两种解决方法: 解决 ...

  9. 20_使用SDL显示BMP图片

    文本的主要内容是:使用SDL显示一张BMP图片,算是为后面的<显示YUV图片>做准备. 为什么是显示BMP图片?而不是显示JPG或PNG图片? 因为SDL内置了加载BMP的API,使用起来 ...

  10. FFmpeg命令行之FFmpeg 采集设备

    在使用 FFmpeg 作为编码器时,可以使用FFmpeg采集本地的音视频采集设备的数据,然后进行编码.封装.传输等操作. 例如,我们可以采集摄像头的图像作为视频,采集麦克风的数据作为音频,然后对采集的 ...