Hdu 2795 线段树 区间最大值的位置.cpp

本文介绍了一个算法,用于在指定尺寸的公告板上合理安排公告的贴附位置,确保公告完整且不超出板面限制。通过构建树状结构并优化查找与更新过程,实现高效的空间利用。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

题意:

  一个board的面积是h*w..在上面贴一些announcement..

  给出告示的宽度ww..可知告示的面积为1*ww..

  问告示是否可以贴上去..可以就输出在board的第几行..不可以就输出-1

 

思路:

  以min(h, n)为长度建一棵树..

  然后每一个的叶子的长度为w..

  找出所有叶子中最大的那个..把announcement贴上去..

  用ma数组存当前区间最大值..

 

Tips:

  查询和更新可以一起写..但是我分开写了..

  然后还有一个要注意的是..树的最大宽度为min(h, n)
  比较的时候是和左右节点大小比..

  ※ 所以modify的时候判断条件也应该是if(x > ma[rt<<1])..

  

 

Code:

View Code
 1 #include <stdio.h>
 2 #include <cstring>
 3 #include <algorithm>
 4 using namespace std;
 5 
 6 const int MAXN = 200010;
 7 int ma[MAXN<<2];
 8 int h, w, n;
 9 
10 void pushup(int rt)
11 {
12     ma[rt] = max(ma[rt<<1], ma[rt<<1|1]);
13 }
14 
15 void creat(int l, int r, int rt)
16 {
17     ma[rt] = w;
18     if(l == r)
19         return;
20     int mid = (l+r)>>1;
21     creat(l, mid, rt<<1);
22     creat(mid+1, r, rt<<1|1);
23 }
24 
25 void modify(int p, int ww, int l, int r, int rt)
26 {
27     if(l == r) {
28         ma[rt] += ww;
29         return;
30     }
31     int mid = (l+r)>>1;
32     if(p <= mid) modify(p, ww, l, mid, rt<<1);
33     else modify(p, ww, mid+1, r, rt<<1|1);
34     pushup(rt);
35 }
36 
37 int query(int x, int L, int R, int rt)
38 {
39     if(L == R) {
40         return L;
41     }
42     int mid = (L+R)>>1;
43     int res = 0;
44     if(x <= ma[rt<<1]) query(x, L, mid, rt<<1);
45     else query(x, mid+1, R, rt<<1|1);
46 }
47 
48 int main()
49 {
50     int i, j;
51     int ww;
52     int ans;
53     while(scanf("%d %d %d", &h, &w, &n) != EOF)
54     {
55         if(h > n) h = n;
56         creat(1, h, 1);
57 
58         for(i = 0; i < n; ++i) {
59             scanf("%d", &ww);
60             if(ww > ma[1]) puts("-1");
61             else {
62                 ans = query(ww, 1, h, 1);
63                 printf("%d\n", ans);
64                 modify(ans, -ww, 1, h, 1);
65             }
66 
67         }
68     }
69     return 0;
70 }

 

 

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

转载于:https://www.cnblogs.com/Griselda/archive/2012/10/11/2719741.html

现在不会TLE了,但还是WA了: #include <bits/stdc++.h> using namespace std; #define int long long const int INF = 0x3f3f3f3f; const int MAX_N = 2e5 + 50; int T, n, m, a[MAX_N]; #define ls(cur) cur << 1 #define rs(cur) cur << 1 | 1 namespace SegmentTree1 { // the case of a[i] <= x long long sum[MAX_N << 2], vtag[MAX_N << 2]; int cnt[MAX_N << 2], stag[MAX_N << 2]; void pushup(int cur) { sum[cur] = sum[ls(cur)] + sum[rs(cur)]; cnt[cur] = cnt[ls(cur)] + cnt[rs(cur)]; } void mark(int cur, int sign, long long val) { sum[cur] *= sign; sum[cur] += cnt[cur] * val; stag[cur] *= sign; vtag[cur] += val; } void pushdown(int cur) { if (stag[cur] != 1) { mark(ls(cur), stag[cur], 0); mark(rs(cur), stag[cur], 0); stag[cur] = 1; } if (vtag[cur]) { mark(ls(cur), 1, vtag[cur]); mark(rs(cur), 1, vtag[cur]); vtag[cur] = 0; } } void build(int cur, int l, int r) { stag[cur] = 1; vtag[cur] = 0; if (l == r) { sum[cur] = cnt[cur] = 0; return ; } int mid = l + r >> 1; build(ls(cur), l, mid); build(rs(cur), mid + 1, r); pushup(cur); } void insert(int cur, int l, int r, int idx, int val) { if (l == r) { sum[cur] = val; cnt[cur] = 1; return ; } pushdown(cur); int mid = l + r >> 1; if (idx <= mid) insert(ls(cur), l, mid, idx, val); else insert(rs(cur), mid + 1, r, idx, val); pushup(cur); } void modify(int cur, int l, int r, int L, int R, int val) { if (L <= l && r <= R) { mark(cur, -1, val); return ; } pushdown(cur); int mid = l + r >> 1; if (L <= mid) modify(ls(cur), l, mid, L, R, val); if (mid + 1 <= R) modify(rs(cur), mid + 1, r, L, R, val); pushup(cur); } long long query(int cur, int l, int r, int L, int R) { if (L <= l && r <= R) return sum[cur]; pushdown(cur); int mid = l + r >> 1; long long res = 0; if (L <= mid) res += query(ls(cur), l, mid, L, R); if (mid + 1 <= R) res += query(rs(cur), mid + 1, r, L, R); return res; } }; namespace SegmentTree2 { // the case of a[i] > x long long sum[MAX_N << 2], tag[MAX_N << 2]; int cnt[MAX_N << 2], mn[MAX_N << 2]; void pushup(int cur) { sum[cur] = sum[ls(cur)] + sum[rs(cur)]; cnt[cur] = cnt[ls(cur)] + cnt[rs(cur)]; mn[cur] = min(mn[ls(cur)], mn[rs(cur)]); } void mark(int cur, long long val) { sum[cur] -= cnt[cur] * val; mn[cur] -= bool(cnt[cur]) * val; tag[cur] += val; } void pushdown(int cur) { if (tag[cur]) { mark(ls(cur), tag[cur]); mark(rs(cur), tag[cur]); tag[cur] = 0; } } void build(int cur, int l, int r, int val[]) { tag[cur] = 0; if (l == r) { sum[cur] = mn[cur] = val[l]; cnt[cur] = 1; return ; } int mid = l + r >> 1; build(ls(cur), l, mid, val); build(rs(cur), mid + 1, r, val); pushup(cur); } void modify(int cur, int l, int r, int L, int R, int val) { if (l == r && mn[cur] <= val) { SegmentTree1::insert(1, 1, n, l, mn[cur]); sum[cur] = cnt[cur] = 0, mn[cur] = INF; return ; } if (L <= l && r <= R && mn[cur] > val) { mark(cur, val); return ; } pushdown(cur); int mid = l + r >> 1; if (L <= mid) modify(ls(cur), l, mid, L, R, val); if (mid + 1 <= R) modify(rs(cur), mid + 1, r, L, R, val); pushup(cur); } long long query(int cur, int l, int r, int L, int R) { if (L <= l && r <= R) return sum[cur]; pushdown(cur); int mid = l + r >> 1; long long res = 0; if (L <= mid) res += query(ls(cur), l, mid, L, R); if (mid + 1 <= R) res += query(rs(cur), mid + 1, r, L, R); return res; } }; signed main() { ios::sync_with_stdio(false); cin.tie(0), cout.tie(0); cin >> T; while (T--) { cin >> n >> m; for (int i = 1; i <= n; i++) cin >> a[i]; SegmentTree1::build(1, 1, n); SegmentTree2::build(1, 1, n, a); for (int opt, l, r, x; m--; ) { cin >> opt >> l >> r; if (opt == 1) { cin >> x; SegmentTree2::modify(1, 1, n, l, r, x); SegmentTree1::modify(1, 1, n, l, r, x); } else { long long ans1 = SegmentTree1::query(1, 1, n, l, r); long long ans2 = SegmentTree2::query(1, 1, n, l, r); cout << ans1 + ans2 << '\n'; } } } return 0; }
最新发布
07-29
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值