[bzoj1558][JSOI2009]等差数列
题目:给定n个数,m个操作,每次给一段区间加一个等差数列或者询问一段区间至少要用多少个等差数列来表示。$n,m\leqslant 10^{5}$
题解:老套路,维护差分数组,修改操作变成了两个单点加和一个区间加。然后我们用线段树维护,合并答案的时候复杂一点,s[0/1][0/1]表示左右端点取不取的答案就行啦。
#include<iostream>
#include<cstdio>
#define MN 100000
using namespace std;
inline int read()
{
int x = , f = ; char ch = getchar();
while(ch < '' || ch > ''){ if(ch == '-') f = -; ch = getchar();}
while(ch >= '' && ch <= ''){x = x * + ch - '';ch = getchar();}
return x * f;
} int n,q,s[MN+];
char op[]; inline void re(int&x,int y){if(y<x) x=y;} struct data
{
int s[],l,r;
data operator + (data y)
{
data c;c.l=l;c.r=y.r;
c.s[]=s[]+y.s[]-(r==y.l);
re(c.s[],s[]+y.s[]);re(c.s[],s[]+y.s[]);
c.s[]=s[]+y.s[]-(r==y.l);
re(c.s[],s[]+y.s[]);re(c.s[],s[]+y.s[]);
c.s[]=s[]+y.s[]-(r==y.l);
re(c.s[],s[]+y.s[]);re(c.s[],s[]+y.s[]);
c.s[]=s[]+y.s[]-(r==y.l);
re(c.s[],s[]+y.s[]);re(c.s[],s[]+y.s[]);
return c;
}
};
struct node{
int l,r,val;data x;
}T[MN*+]; void pushdown(int x)
{
int l=x<<,r=x<<|;
T[l].val+=T[x].val;T[r].val+=T[x].val;
T[l].x.l+=T[x].val;T[l].x.r+=T[x].val;
T[r].x.l+=T[x].val;T[r].x.r+=T[x].val;
T[x].val=;
} void build(int x,int l,int r)
{
if((T[x].l=l)==(T[x].r=r))
{
T[x].x.s[]=;T[x].x.l=T[x].x.r=s[l];
T[x].x.s[]=T[x].x.s[]=T[x].x.s[]=;
return;
}
int mid=l+r>>;
build(x<<,l,mid);build(x<<|,mid+,r);
T[x].x=T[x<<].x+T[x<<|].x;
} data query(int x,int l,int r)
{
//cout<<"query"<<x<<" "<<l<<" "<<r<<" "<<T[x].l<<" "<<T[x].r<<endl;
if(T[x].l==l&&T[x].r==r) return T[x].x;
if(T[x].val) pushdown(x);
int mid=T[x].l+T[x].r>>;
if(r<=mid) return query(x<<,l,r);
else if(l>mid) return query(x<<|,l,r);
else return query(x<<,l,mid)+query(x<<|,mid+,r);
} void modify(int x,int l,int r,int ad)
{
if(T[x].l==l&&T[x].r==r)
{
T[x].val+=ad;T[x].x.l+=ad;T[x].x.r+=ad;
return;
}
if(T[x].val) pushdown(x);
int mid=T[x].l+T[x].r>>;
if(r<=mid) modify(x<<,l,r,ad);
else if(l>mid) modify(x<<|,l,r,ad);
else modify(x<<,l,mid,ad),modify(x<<|,mid+,r,ad);
T[x].x=T[x<<].x+T[x<<|].x;
} int main()
{
n=read();
for(int i=;i<=n;i++) s[i]=read();
for(int i=;i<n;i++) s[i]=s[i+]-s[i];
build(,,n-);
for(q=read();q;q--)
{
scanf("%s",op);int l=read(),r=read();
if(op[]=='B') l==r?puts(""):printf("%d\n",query(,l,r-).s[]);
else
{
int a=read(),b=read();
if(l!=) modify(,l-,l-,a);
if(l!=r) modify(,l,r-,b);
if(r!=n) modify(,r,r,-(a+(r-l)*b));
}
}
return ;
}
[bzoj1558][JSOI2009]等差数列的更多相关文章
- BZOJ1558 [JSOI2009]等差数列 【线段树】
题目链接 BZOJ1558 题解 等差数列,当然是差分一下 差分值相同的连续位置形成等差数列,我们所选的两个等差数列之间可以有一个位置舍弃 例如: \(1 \; 2 \; 3 \; 6 \; 8 \; ...
- 洛谷P4243/bzoj1558 [JSOI2009]等差数列(线段树维护差分+爆炸恶心的合并)
题面 首先感谢这篇题解,是思路来源 看到等差数列,就会想到差分,又有区间加,很容易想到线段树维护差分.再注意点细节,\(A\)操作完美解决 然后就是爆炸恶心的\(B\)操作,之前看一堆题解的解释都不怎 ...
- 【BZOJ1558】等差数列(线段树)
[BZOJ1558]等差数列(线段树) 题面 BZOJ 题解 可以说这道题已经非常毒瘤了 怎么考虑询问操作? 如果直接将一段数分解为等差数列? 太麻烦了.... 考虑相邻的数做差, 这样等差数列变为了 ...
- bzoj 1558: [JSOI2009]等差数列
Description Solution 把原数组变为差分数组,然后剩下的就十分显然了 区间查询用线段树维护 修改操作就是区间加法和两个单点修改 一个等差数列实际上就是 开头一个数字+数值相等的一段 ...
- BZOJ.1558.[JSOI2009]等差数列(线段树 差分)
BZOJ 洛谷 首先可以把原序列\(A_i\)转化成差分序列\(B_i\)去做. 这样对于区间加一个等差数列\((l,r,a_0,d)\),就可以转化为\(B_{l-1}\)+=\(a_0\),\(B ...
- [JSOI2009]等差数列
链接:http://www.lydsy.com/JudgeOnline/problem.php?id=1558 题解: 考虑这么用线段树进行维护,由于他有区间修改等差数列 很容易想到可以用差分数组来维 ...
- JSOI2009 等差数列 和 算术天才⑨与等差数列 和 CH4302 Interval GCD
等差数列 为了检验学生的掌握情况,jyy布置了一道习题:给定一个长度为N(1≤N≤100,000)的数列,初始时第i个数为vi(vi是整数,−100,000≤vi≤100,000),学生们要按照jyy ...
- luogu P4243 [JSOI2009]等差数列 题解
前言: 这题真ex... 强烈谴责在题解里面放毒瘤题链接的屑出题人! 吐 ️ 解析: 这题分成两步走. 首先,既然题目中的修改操作是区间加等差数列,那么就容易想到在差分数组上进行操作. 然后就是相当恶 ...
- 线段树 by yyb
线段树 by yyb Type1 维护特殊信息 1.[洛谷1438]无聊的数列 维护一个数列,两种操作 1.给一段区间加上一个等差数列 2.单点询问值 维护等差数列 不难发现,等差数列可以写成\(ad ...
随机推荐
- signalR 消息推送
业务情景一:上传报表,上传excel.如果excel的数据量很大,上万条,上十万条数据,那么这个上传请求必然是个耗时请求.用户上传之后,很关心上传的进度和结果. 业务情景二:站内消息提醒,实时有效地接 ...
- 【微软大法好】VS Tools for AI全攻略
大家都知道微软在Connect();17大会上发布了VS Tools for AI,旨在提升Visual Studio和VSCode对日益增长的深度学习需求的体验.看了一圈,网上似乎没有一个完整的中文 ...
- 测试驱动开发实践3————从testList开始
[内容指引] 运行单元测试: 装配一条数据: 模拟更多数据测试列表: 测试无搜索列表: 测试标准查询: 测试高级查询. 一.运行单元测试 我们以文档分类(Category)这个领域类为例,示范如何通过 ...
- angular4学习笔记整理(二)angular4的路由使用
这章说一下angular的路由 先说angular路由怎么引入,一开始new出来的angular项目它路由帮你配好了,但看要看app.module.ts里面 1.首先最上面要引入路由模块 import ...
- linux下的Shell编程(5)循环
Shell Script中的循环有下面几种格式: while [ cond1 ] && { || } [ cond2 ] -; do - done for var in -; do - ...
- Django之views系统
Django的View(视图)简介 一个视图函数(类),简称视图,是一个简单的Python 函数(类),它接受Web请求并且返回Web响应. 响应可以是一张网页的HTML内容,一个重定向,一个404错 ...
- ActiveMQ学习系列(二)----生产者客户端(java)
上文主要简单地将activeMq搭建了起来,并且可以用web console去登录查看相关的后台功能. 本文将学习如何用java语言实现一个生产者客户端,主要参考了以下链接: http://activ ...
- Git撤销commit消息保留修改
有时候commit后发现commit信息错了或者是添加了不想commit的内容,但还没有push到远程仓库 这个时候 git reset --soft [commit_id] 就可以回滚到某一个com ...
- hive:数据库“行专列”操作---使用collect_set/collect_list/collect_all & row_number()over(partition by 分组字段 [order by 排序字段])
方案一:请参考<数据库“行专列”操作---使用row_number()over(partition by 分组字段 [order by 排序字段])>,该方案是sqlserver,orac ...
- POJ-2421 Constructing Roads---确定部分边的MST
题目链接: https://vjudge.net/problem/POJ-2421 题目大意: 还是给你n个点,然后求最小生成树.特殊之处在于有一些点之间已经连上了边. 思路: 和POJ-1751一样 ...