前言

spoj需要翻墙注册,比较麻烦,大家就在luogu评测吧

题目大意:

$n$ 个数,$q$ 次操作

操作$0 _ x_ y$把$A_x$ 修改为$y$

操作$1 _ l _r$询问区间$[l, r]$ 的最大子段和

## 思路
###**维护一个区间的$sum,lmax,rmax,max$**
###**就是区间和,左端开头的最大字段和,右端开头的最大字段和和区间内的最大字段和**
###**他们都说转移难,我咋感觉是询问难呢**
###**转移的话**
####**①max 就是右边,左边,和交界处的最大值,取个max**
####**②lmax和rmax类似,就是有两种可能,一种是只在一个孩子内,一种是两个孩子内(完全包含一个孩子)**
####**③sum不必说(其实就是为了求lmax和rmax的)**

查询的话,当然不能加起来啦,如果不连续不就GG了

①如果全部在一个孩子内,直接去查询那个孩子就好

②如果横跨两个孩子的话,还要讨论一下交界处

看着很明白,也许是我太菜了不太会递归吧

我居然能忘记维护sum,也是醉了

代码

#include <iostream>
#include <cstdio>
#include <cstring>
#include <algorithm>
#include <cmath>
#include <bitset>
#define ll long long
#define ls rt<<1
#define rs rt<<1|1
using namespace std;
const int maxn = 50007;
const int maxm = 250007;
const int inf = 0x3f3f3f3f; int n, m, a[maxn],tot[maxn];
struct node {
int l, r;
int lk, rk, k, sum;
} e[maxm]; int read() {
int x = 0, f = 1; char s = getchar();
for (; s > '9' || s < '0'; s = getchar()) if (s == '-') f = -1;
for (; s <= '9' && s >= '0'; s = getchar()) x = x * 10 + s - '0';
return x * f;
} int max_2(int a, int b, int c) {
return max(max(a, b), c);
} void pushup(int rt) {
e[rt].k = max_2(e[ls].k, e[rs].k, e[ls].rk + e[rs].lk);
e[rt].lk = max(e[ls].sum + e[rs].lk, e[ls].lk);
e[rt].rk = max(e[ls].rk + e[rs].sum, e[rs].rk);
e[rt].sum= e[ls].sum +e[rs].sum;
} void build(int l, int r, int rt) {
e[rt].l = l, e[rt].r = r, e[rt].sum = tot[r] - tot[l-1];
if (l == r) {
e[rt].k = e[rt].lk = e[rt].rk = a[l];
return;
}
int mid = (l + r) >> 1;
build(l, mid, ls);
build(mid + 1, r, rs);
pushup(rt);
} void update(int L, int k, int rt) {
if (e[rt].l == e[rt].r) {
e[rt].sum=k;
e[rt].k = e[rt].lk = e[rt].rk = k;
return;
}
int mid = (e[rt].l + e[rt].r) >> 1;
if (L <= mid) update(L, k, ls);
else update(L, k, rs);
pushup(rt);
}
void debug() {
printf("debug\n");
printf(" %d\n", e[1].k);
printf(" %d %d\n", e[2].k, e[3].k );
printf(" %d %d %d %d\n", e[4].k, e[5].k, e[6].k, e[7].k );
} node query(int L, int R, int rt) {
if(L<=e[rt].l&&e[rt].r<=R) {
return e[rt];
}
int mid = (e[rt].l + e[rt].r) >> 1;
if(L <= mid && R > mid) {
node x=query(L, R, ls),y=query(L, R, rs);
node tmp={}; tmp.k = max_2(x.k, y.k, x.rk + y.lk);
tmp.lk =max(x.sum + y.lk, x.lk);
tmp.rk =max(x.rk + y.sum, y.rk);
return tmp;
}
if (L <= mid) return query(L, R, ls);
if (R > mid) return query(L, R, rs);
} int main() {
//freopen("in.txt","r",stdin);
n = read();
for (int i = 1; i <= n; ++i) {
a[i] = read();
tot[i]=tot[i-1]+a[i];
}
build(1, n, 1);
m = read();
for (; m--;) {
int tmp = read(), a = read(), b = read();
if (tmp) {
printf("%d\n",max_2(query(a, b, 1).k,query(a,b,1).lk,query(a,b,1).rk) );
} else {
update(a, b, 1);
}
}
//debug();
return 0;
}

最大子段和SP1716GSS3 线段树的更多相关文章

  1. 51nod 1081 子段求和(线段树 | 树状数组 | 前缀和)

    题目链接:子段求和 题意:n个数字序列,m次询问,每次询问从第p个开始L长度序列的子段和为多少. 题解:线段树区间求和 | 树状数组区间求和 线段树: #include <cstdio> ...

  2. 洛谷 P4513 小白逛公园-区间最大子段和-分治+线段树区间合并(单点更新、区间查询)

    P4513 小白逛公园 题目背景 小新经常陪小白去公园玩,也就是所谓的遛狗啦… 题目描述 在小新家附近有一条“公园路”,路的一边从南到北依次排着nn个公园,小白早就看花了眼,自己也不清楚该去哪些公园玩 ...

  3. codevs3981动态最大子段和(线段树)

    3981 动态最大子段和  时间限制: 1 s  空间限制: 128000 KB  题目等级 : 钻石 Diamond     题目描述 Description 题目还是简单一点好... 有n个数,a ...

  4. codevs 3981 动态最大子段和(线段树)

    题目传送门:codevs 3981 动态最大子段和 题目描述 Description 题目还是简单一点好... 有n个数,a[1]到a[n]. 接下来q次查询,每次动态指定两个数l,r,求a[l]到a ...

  5. BZOJ5142: [Usaco2017 Dec]Haybale Feast 线段树或二分答案

    Description Farmer John is preparing a delicious meal for his cows! In his barn, he has NN haybales ...

  6. BZOJ1135:[POI2009]Lyz(线段树,Hall定理)

    Description 初始时滑冰俱乐部有1到n号的溜冰鞋各k双.已知x号脚的人可以穿x到x+d的溜冰鞋. 有m次操作,每次包含两个数ri,xi代表来了xi个ri号脚的人.xi为负,则代表走了这么多人 ...

  7. SPOJ GSS3-Can you answer these queries III-分治+线段树区间合并

    Can you answer these queries III SPOJ - GSS3 这道题和洛谷的小白逛公园一样的题目. 传送门: 洛谷 P4513 小白逛公园-区间最大子段和-分治+线段树区间 ...

  8. Luogu P2572 [SCOI2010]序列操作 线段树。。

    咕咕了...于是借鉴了小粉兔的做法ORZ... 其实就是维护最大子段和的线段树,但上面又多了一些操作....QWQ 维护8个信息:1/0的个数(sum),左/右边起1/0的最长长度(ls,rs),整段 ...

  9. HDU 6638 - Snowy Smile 线段树区间合并+暴力枚举

    HDU 6638 - Snowy Smile 题意 给你\(n\)个点的坐标\((x,\ y)\)和对应的权值\(w\),让你找到一个矩形,使这个矩阵里面点的权值总和最大. 思路 先离散化纵坐标\(y ...

随机推荐

  1. RSA原理说明

    长度,建议至少1024.模数n(常取默认65537)两边都要用. 指数e,和n一起就是公钥. 指数d,和n一起就是私钥. 质数p和q用于生成密钥对,然后就丢弃不公开. 一.密钥对的生成步骤 1.随机选 ...

  2. MySQL如何开启慢查询

    一 简介 开启慢查询日志,可以让MySQL记录下查询超过指定时间的语句,通过定位分析性能的瓶颈,才能更好的优化数据库系统的性能.   二 参数说明 slow_query_log 慢查询开启状态 slo ...

  3. NYOJ 275 队花的烦恼一

    队花的烦恼一 时间限制:3000 ms  |  内存限制:65535 KB 难度:1 描写叙述 ACM队的队花C小+常常抱怨:"C语言中的格式输出中有十六.十.八进制输出,然而却没有二进制输 ...

  4. iOS10网络权限数据

    参考地址:1.http://www.cocoachina.com/ios/20180723/24274.html   https://blog.csdn.net/wang_bo_justone/art ...

  5. [py]你真的了解多核处理器吗? 了解多线程

    越来越多的人搞爬虫,设计到多线程爬取, 还有一些机器学习的一些模块也需要这玩意, 感觉自己不会逼格不高. 抽时间赶紧玩一玩这东西, 希望提高对软件的认知和归属感,不要太傻. cpu内部架构参考 你知道 ...

  6. POJ3169:Layout(差分约束)

    http://poj.org/problem?id=3169 题意: 一堆牛在一条直线上按编号站队,在同一位置可以有多头牛并列站在一起,但编号小的牛所占的位置不能超过编号大的牛所占的位置,这里用d[i ...

  7. PAT 1045 Favorite Color Stripe[dp][难]

    1045 Favorite Color Stripe (30)(30 分) Eva is trying to make her own color stripe out of a given one. ...

  8. [LeetCode] 561. Array Partition I_Easy tag: Sort

    Given an array of 2n integers, your task is to group these integers into n pairs of integer, say (a1 ...

  9. Windows多线程基础

    进程与线程基础 程序: 计算机指令的集合,以文件的形式存储在磁盘上 进程: 正在运行是程序实例,以是一个程序在其自身的地址空间的一次执行活动.进程有一个进程管理的内核对象和地址空间组成. 线程: 程序 ...

  10. iOS 新浪微博-1.0框架搭建

    项目搭建 1.新建一个微博的项目,去掉屏幕旋转 2.设置屏幕方向-->只有竖向 3.使用代码构建UI,不使用storyboard 4.配置图标AppIcon和LaunchImage 将微博资料的 ...