uva 1400 - Ray, Pass me the dishes!(线段树)

2024-06-05 02:32
文章标签 pass 线段 ray 1400 uva dishes

本文主要是介绍uva 1400 - Ray, Pass me the dishes!(线段树),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

题目链接:uva 1400 - "Ray, Pass me the dishes!"

题目大意:给定一个长度为n个整数序列,对m次询问作出回答,对于每次询问(a,b),找到两个下标x,y使得x到y的连续和为区间a,b中最大的连续和,如果存在多解优先x小,然后y小。

解题思路:线段树,对于每个节点维护三个线段值:

  • max_sub:区间连续最大和
  • max_prefix:区间连续前缀最大和
  • max_suffix:区间连续后缀最大和
    建树的过程维护三个值,查询时只需考虑左右子节点的max_sub,以及两边max_suffix+max_prefix
#include <cstdio>
#include <cstring>
#include <algorithm>using namespace std;
#define lson(x) ((x)<<1)
#define rson(x) (((x)<<1)+1)
typedef long long ll;
const int maxn = 500005;
const ll INF = 0x3f3f3f3f3f3f3f3f;struct segment {int l, r;ll val;segment (int l = 0, int r = 0, ll val = 0) {this->set(l, r, val);}void set (int l, int r, ll val) {this->l = l;this->r = r;this->val = val;}segment operator + (const segment& u) {return segment(min(l, u.l), max(r, u.r), val + u.val);}bool operator < (const segment& u) const {if (val != u.val)return val < u.val;if (l != u.l)return l > u.l;return r > u.r;}
};struct Node {int l, r;segment max_sub, max_prefix, max_suffix;
} node[4*maxn];
int N, M;
ll arr[maxn], s[maxn];Node seg_push (Node a, Node b) {Node ret;ll suml = s[a.r] - s[a.l-1];ll sumr = s[b.r] - s[b.l-1];ret.l = a.l;ret.r = b.r;ret.max_sub = max(a.max_suffix + b.max_prefix, max(a.max_sub, b.max_sub));ret.max_prefix = max(a.max_prefix, segment(a.l, a.r, suml) + b.max_prefix);ret.max_suffix = max(b.max_suffix, a.max_suffix + segment(b.l, b.r, sumr));return ret;
}void build_segTree (int root, int l, int r) {if (l == r) {node[root].l = node[root].r = l;node[root].max_sub.set(l, r, (ll)arr[l]);node[root].max_prefix.set(l, r, (ll)arr[l]);node[root].max_suffix.set(l, r, (ll)arr[l]);return;}int mid = (l + r) / 2;build_segTree(lson(root), l, mid);build_segTree(rson(root), mid + 1, r);node[root] = seg_push(node[lson(root)], node[rson(root)]);
}Node query (int root, int l, int r) {if (l <= node[root].l && r >= node[root].r)return node[root];int mid = (node[root].l + node[root].r) / 2;if (l <= mid && r > mid)return seg_push(query(lson(root), l, r), query(rson(root), l, r));else if (r <= mid)return query(lson(root), l, r);elsereturn query(rson(root), l, r);
}int main () {int cas = 1;while (scanf("%d%d", &N, &M) == 2) {s[0] = 0;for (int i = 1; i <= N; i++) {scanf("%lld", &arr[i]);s[i] = s[i-1] + arr[i];}build_segTree(1, 1, N);printf("Case %d:\n", cas++);int l, r;for (int i = 0; i < M; i++) {scanf("%d%d", &l, &r);Node u = query(1, l, r);printf("%d %d\n", u.max_sub.l, u.max_sub.r);}}return 0;
}

这篇关于uva 1400 - Ray, Pass me the dishes!(线段树)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1031821

相关文章

poj3468(线段树成段更新模板题)

题意:包括两个操作:1、将[a.b]上的数字加上v;2、查询区间[a,b]上的和 下面的介绍是下解题思路: 首先介绍  lazy-tag思想:用一个变量记录每一个线段树节点的变化值,当这部分线段的一致性被破坏我们就将这个变化值传递给子区间,大大增加了线段树的效率。 比如现在需要对[a,b]区间值进行加c操作,那么就从根节点[1,n]开始调用update函数进行操作,如果刚好执行到一个子节点,

hdu1394(线段树点更新的应用)

题意:求一个序列经过一定的操作得到的序列的最小逆序数 这题会用到逆序数的一个性质,在0到n-1这些数字组成的乱序排列,将第一个数字A移到最后一位,得到的逆序数为res-a+(n-a-1) 知道上面的知识点后,可以用暴力来解 代码如下: #include<iostream>#include<algorithm>#include<cstring>#include<stack>#in

hdu1689(线段树成段更新)

两种操作:1、set区间[a,b]上数字为v;2、查询[ 1 , n ]上的sum 代码如下: #include<iostream>#include<algorithm>#include<cstring>#include<stack>#include<queue>#include<set>#include<map>#include<stdio.h>#include<stdl

uva 10055 uva 10071 uva 10300(水题两三道)

情歌两三首,水题两三道。 好久没敲代码了为暑假大作战热热身。 uva 10055 Hashmat the Brave Warrior 求俩数相减。 两个debug的地方,一个是longlong,一个是输入顺序。 代码: #include<stdio.h>int main(){long long a, b;//debugwhile(scanf("%lld%lld", &

poj 3259 uva 558 Wormholes(bellman最短路负权回路判断)

poj 3259: 题意:John的农场里n块地,m条路连接两块地,w个虫洞,虫洞是一条单向路,不但会把你传送到目的地,而且时间会倒退Ts。 任务是求你会不会在从某块地出发后又回来,看到了离开之前的自己。 判断树中是否存在负权回路就ok了。 bellman代码: #include<stdio.h>const int MaxN = 501;//农场数const int

poj 2349 Arctic Network uva 10369(prim or kruscal最小生成树)

题目很麻烦,因为不熟悉最小生成树的算法调试了好久。 感觉网上的题目解释都没说得很清楚,不适合新手。自己写一个。 题意:给你点的坐标,然后两点间可以有两种方式来通信:第一种是卫星通信,第二种是无线电通信。 卫星通信:任何两个有卫星频道的点间都可以直接建立连接,与点间的距离无关; 无线电通信:两个点之间的距离不能超过D,无线电收发器的功率越大,D越大,越昂贵。 计算无线电收发器D

uva 10387 Billiard(简单几何)

题意是一个球从矩形的中点出发,告诉你小球与矩形两条边的碰撞次数与小球回到原点的时间,求小球出发时的角度和小球的速度。 简单的几何问题,小球每与竖边碰撞一次,向右扩展一个相同的矩形;每与横边碰撞一次,向上扩展一个相同的矩形。 可以发现,扩展矩形的路径和在当前矩形中的每一段路径相同,当小球回到出发点时,一条直线的路径刚好经过最后一个扩展矩形的中心点。 最后扩展的路径和横边竖边恰好组成一个直

uva 10061 How many zero's and how many digits ?(不同进制阶乘末尾几个0)+poj 1401

题意是求在base进制下的 n!的结果有几位数,末尾有几个0。 想起刚开始的时候做的一道10进制下的n阶乘末尾有几个零,以及之前有做过的一道n阶乘的位数。 当时都是在10进制下的。 10进制下的做法是: 1. n阶位数:直接 lg(n!)就是得数的位数。 2. n阶末尾0的个数:由于2 * 5 将会在得数中以0的形式存在,所以计算2或者计算5,由于因子中出现5必然出现2,所以直接一

uva 568 Just the Facts(n!打表递推)

题意是求n!的末尾第一个不为0的数字。 不用大数,特别的处理。 代码: #include <stdio.h>const int maxn = 10000 + 1;int f[maxn];int main(){#ifdef LOCALfreopen("in.txt", "r", stdin);#endif // LOCALf[0] = 1;for (int i = 1; i <=

uva 575 Skew Binary(位运算)

求第一个以(2^(k+1)-1)为进制的数。 数据不大,可以直接搞。 代码: #include <stdio.h>#include <string.h>const int maxn = 100 + 5;int main(){char num[maxn];while (scanf("%s", num) == 1){if (num[0] == '0')break;int len =