hdu - 1823 - Luck and Love(线段树)

2023-11-03 12:49
文章标签 hdu 线段 love luck 1823

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

题意:Wiskey招女友,每个女生看其身高、活泼度和缘分值。现在执行两种操作,1、I,加入一位女生的身高,活泼度和缘分值;2、Q,查询身高在H1, H2之间,活泼度在A1, A2之间的女生的最高缘分值。

题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=1823

——>>查询某个区间的最值,若是一维,可用RMQ解法。。也可用线段树解法。。

现在要查身高限一个区间,活泼度限一个区间,是一个二维的情景。。将线段树扩至二维。。时间复杂度:O(N+M*log(N)^2)。。

——>>坑1:G++的BUG。。大哭。。以G++提交数次皆WA。。改交C++即过。。有朋友指出是代码触发了未定义行为,也有朋友说是因为G++的O2 BUG。。

——>>坑2:题目中说是1位小数,那么,处理方法用scanf("%d.%d", &A1, &A2),再进行A1 * 10 + A2,会比用scanf("%lf", &A),再进行(int)(A * 10)更精确(自己测下0.0到0.9就可以发现)。。可是,用第一种处理方式却会WA,偏偏要用第二种不那么精确的处理方式才会AC。。大哭

第一道二维线段树题目,做到想哭了。。

#include <cstdio>
#include <algorithm>using namespace std;#define lc (o<<1)
#define rc ((o<<1)|1)const int maxh = 100 + 10;
const int maxa = 1000 + 10;
int mmax[maxh<<2][maxa<<2];void build_A(int O, int o, int L, int R) {      //活泼度建树mmax[O][o] = -1;if(L == R) return;int M = (L + R) >> 1;build_A(O, lc, L, M);build_A(O, rc, M+1, R);
}void build_H(int o, int L, int R) {     //身高建树build_A(o, 1, 0, 1000);if(L == R) return;int M = (L + R) >> 1;build_H(lc, L, M);build_H(rc, M+1, R);
}void Insert_A(int O, int o, int L, int R, int A, int D) {       //根据活泼度建树if(L == R) {mmax[O][o] = max(mmax[O][o], D);return;}int M = (L + R) >> 1;if(A <= M) Insert_A(O, lc, L, M, A, D);else Insert_A(O, rc, M+1, R, A, D);mmax[O][o] = max(mmax[O][lc], mmax[O][rc]);
}void Insert(int o, int L, int R, int H, int A, int D) {Insert_A(o, 1, 0, 1000, A, D);if(L == R) return;int M = (L + R) >> 1;if(H <= M) Insert(lc, L, M, H, A, D);else Insert(rc, M+1, R, H, A, D);
}int query_A(int O, int o, int L, int R, int A1, int A2) {       //根据活泼度查询if(A1 <= L && R <= A2) return mmax[O][o];int M = (L + R) >> 1;int Max1 = -1, Max2 = -1;if(A1 <= M) Max1 = query_A(O, lc, L, M, A1, A2);if(A2 > M) Max2 = query_A(O, rc, M+1, R, A1, A2);return (Max1 > Max2) ? Max1 : Max2;
}int query(int o, int L, int R, int H1, int H2, int A1, int A2) {if(H1 <= L && R <= H2) return query_A(o, 1, 0, 1000, A1, A2);int M = (L + R) >> 1;int Max1 = -1, Max2 = -1;if(H1 <= M) Max1 = query(lc, L, M, H1, H2, A1, A2);if(H2 > M) Max2 = query(rc, M+1, R, H1, H2, A1, A2);return (Max1 > Max2) ? Max1 : Max2;
}int main()
{int M;char op;while(scanf("%d", &M) == 1 && M) {build_H(1, 100, 200);for(int i = 0; i < M; i++) {getchar();op = getchar();if(op == 'I') {
//                int H, A, A1, A2, D, D1, D2;
//                scanf("%d %d.%d %d.%d", &H, &A1, &A2, &D1, &D2);
//                A = A1 * 10 + A2;
//                D = D1 * 10 + D2;int H, A, D;double AA, DD;scanf("%d%lf%lf", &H, &AA, &DD);A = (int)(AA * 10);D = (int)(DD * 10);Insert(1, 100, 200, H, A, D);}else {
//                int H1, H2, A[6];
//                scanf("%d %d %d.%d %d.%d", &H1, &H2, A+2, A+3, A+4, A+5);
//                A[0] = A[2] * 10 + A[3];
//                A[1] = A[4] * 10 + A[5];int H1, H2, A1, A2;double AA, BB;scanf("%d%d%lf%lf", &H1, &H2, &AA, &BB);A1 = (int)(AA * 10);A2 = (int)(BB * 10);if(A1 > A2) swap(A1, A2);if(H1 > H2) swap(H1, H2);
//                if(A[0] > A[1]) swap(A[0], A[1]);
//                int ret = query(1, 100, 200, H1, H2, A[0], A[1]);int ret = query(1, 100, 200, H1, H2, A1, A2);ret != -1 ? printf("%.1f\n", ret/10.0) : puts("-1");}}}return 0;
}


这篇关于hdu - 1823 - Luck and Love(线段树)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

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

usaco 1.3 Mixing Milk (结构体排序 qsort) and hdu 2020(sort)

到了这题学会了结构体排序 于是回去修改了 1.2 milking cows 的算法~ 结构体排序核心: 1.结构体定义 struct Milk{int price;int milks;}milk[5000]; 2.自定义的比较函数,若返回值为正,qsort 函数判定a>b ;为负,a<b;为0,a==b; int milkcmp(const void *va,c

poj 3974 and hdu 3068 最长回文串的O(n)解法(Manacher算法)

求一段字符串中的最长回文串。 因为数据量比较大,用原来的O(n^2)会爆。 小白上的O(n^2)解法代码:TLE啦~ #include<stdio.h>#include<string.h>const int Maxn = 1000000;char s[Maxn];int main(){char e[] = {"END"};while(scanf("%s", s) != EO

hdu 2093 考试排名(sscanf)

模拟题。 直接从教程里拉解析。 因为表格里的数据格式不统一。有时候有"()",有时候又没有。而它也不会给我们提示。 这种情况下,就只能它它们统一看作字符串来处理了。现在就请出我们的主角sscanf()! sscanf 语法: #include int sscanf( const char *buffer, const char *format, ... ); 函数sscanf()和

hdu 2602 and poj 3624(01背包)

01背包的模板题。 hdu2602代码: #include<stdio.h>#include<string.h>const int MaxN = 1001;int max(int a, int b){return a > b ? a : b;}int w[MaxN];int v[MaxN];int dp[MaxN];int main(){int T;int N, V;s

hdu 1754 I Hate It(线段树,单点更新,区间最值)

题意是求一个线段中的最大数。 线段树的模板题,试用了一下交大的模板。效率有点略低。 代码: #include <stdio.h>#include <string.h>#define TREE_SIZE (1 << (20))//const int TREE_SIZE = 200000 + 10;int max(int a, int b){return a > b ? a :

hdu 1166 敌兵布阵(树状数组 or 线段树)

题意是求一个线段的和,在线段上可以进行加减的修改。 树状数组的模板题。 代码: #include <stdio.h>#include <string.h>const int maxn = 50000 + 1;int c[maxn];int n;int lowbit(int x){return x & -x;}void add(int x, int num){while

hdu 3790 (单源最短路dijkstra)

题意: 每条边都有长度d 和花费p,给你起点s 终点t,要求输出起点到终点的最短距离及其花费,如果最短距离有多条路线,则输出花费最少的。 解析: 考察对dijkstra的理解。 代码: #include <iostream>#include <cstdio>#include <cstdlib>#include <algorithm>#include <cstrin