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

2024-06-01 20:08
文章标签 pass 线段 ray 1400 uva dishes

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

UVA 1400 - "Ray, Pass me the dishes!"

题目链接

题意:给定一个序列,每次询问一个[L,R]区间,求出这个区间的最大连续子序列和

思路:线段树,每个节点维护3个值,最大连续子序列,最大连续前缀序列,最大连续后缀序列,那么每次pushup的时候,根据这3个序列去拼凑得到新的一个结点即可

代码:

#include <cstdio>
#include <cstring>
#include <algorithm>
using namespace std;#define lson(x) ((x<<1) + 1)
#define rson(x) ((x<<1) + 2)
#define MP(a, b) make_pair(a, b)typedef long long ll;
typedef pair<int, int> Point;const int N = 500005;int n, m;
ll a[N], sum[N];struct Node {int l, r;int prex, sufx;Point sub;
} node[4 * N];ll get(Point x) {return sum[x.second] - sum[x.first - 1];
}bool Max(Point a, Point b) {long long sa = get(a);long long sb = get(b);if (sa != sb) return sa > sb;return a < b;
}Point Maxsub(Node a, Node b) {Point ans;if (Max(a.sub, b.sub)) ans = a.sub;else ans = b.sub;if (Max(MP(a.sufx, b.prex), ans)) ans = MP(a.sufx, b.prex);return ans;
}int Maxpre(Node a, Node b) {Point ans = MP(a.l, a.prex);if (Max(MP(a.l, b.prex), ans)) ans = MP(a.l, b.prex);return ans.second;
}int Maxsuf(Node a, Node b) {Point ans = MP(b.sufx, b.r);if (Max(MP(a.sufx, b.r), ans)) ans = MP(a.sufx, b.r);return ans.first;
}Node pushup(Node a, Node b) {Node ans;ans.l = a.l; ans.r = b.r;ans.sub = Maxsub(a, b);ans.prex = Maxpre(a, b);ans.sufx = Maxsuf(a, b);return ans;
}void build(int l, int r, int x) {if (l == r) {node[x].l = l; node[x].r = r;node[x].prex = node[x].sufx = l;node[x].sub = MP(l, l);return ;}int mid = (l + r) / 2;build(l, mid, lson(x));build(mid + 1, r, rson(x));node[x] = pushup(node[lson(x)], node[rson(x)]);
}Node Query(int l, int r, int x) {if (l <= node[x].l && r >= node[x].r)return node[x];int mid = (node[x].l + node[x].r) / 2;Node ans;if (l <= mid && r > mid)ans = pushup(Query(l, r, lson(x)), Query(l, r, rson(x)));else if (l <= mid) ans = Query(l, r, lson(x));else if (r > mid) ans = Query(l, r, rson(x));return ans;
}int main() {int cas = 0;while (~scanf("%d%d", &n, &m)) {for (int i = 1; i <= n; i++) {scanf("%lld", &a[i]);sum[i] = sum[i - 1] + a[i];}build(1, n, 0);printf("Case %d:\n", ++cas);int a, b;while (m--) {scanf("%d%d", &a, &b);Node ans = Query(a, b, 0);printf("%d %d\n", ans.sub.first, ans.sub.second);}}return 0;
}


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



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

相关文章

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 =