HDU 5700——区间交——————【线段树+枚举】

区间交

Time Limit: 8000/4000 MS (Java/Others)    Memory Limit: 65536/65536 K (Java/Others)
Total Submission(s): 567    Accepted Submission(s): 279

Problem Description

小A有一个含有n个非负整数的数列与m个区间。每个区间可以表示为li,ri。

它想选择其中k个区间, 使得这些区间的交的那些位置所对应的数的和最大。

例如样例中,选择[2,5]与[4,5]两个区间就可以啦。

Input

多组测试数据

第一行三个数n,k,m(1≤n≤100000,1≤k≤m≤100000)。

接下来一行n个数ai,表示lyk的数列(0≤ai≤109)。

接下来m行,每行两个数li,ri,表示每个区间(1≤li≤ri≤n)。

Output

一行表示答案

Sample Input

5 2 3
1 2 3 4 6
4 5
2 5
1 4

Sample Output

10

Source

2016"百度之星" - 初赛(Astar Round2B)

解题思路:首先排序右端点从小到大(也可以排序左端点,也可以从大到小排序,看怎么处理了),然后枚举右端点(保证所枚举的那个端点最少有k个区间可以覆盖)作为所求的交区间的右端点,这时候需要求出交区间的左端点,我们可以知道,右端点确定下,如果左端点越靠左,这个区间的范围约大。为了保证所交区间有k个,我们需要找到第k小的左端点,为了保证我枚举的右端点肯定是交区间的右端点,我们必须边枚举,边单点更新左端点。

#include<stdio.h>
#include<algorithm>
#include<string.h>
#include<math.h>
#include<string>
#include<iostream>
#include<queue>
#include<stack>
#include<map>
#include<vector>
#include<set>
using namespace std;
typedef long long LL;
#define mid (L+R)/2
#define lson rt*2,L,mid
#define rson rt*2+1,mid+1,R
#pragma comment(linker, "/STACK:102400000,102400000")
const int maxn = 1e5+300;
const int INF = 0x3f3f3f3f;
typedef long long  LL;
typedef unsigned long long ULL;
LL presum[maxn];
struct Interval{
    int l, r;
}intervals[maxn];
struct Seg{
    int cover;
}segs[maxn*4];
bool cmp(Interval a, Interval b){
    return a.r < b.r;           //
}
void PushUp(int rt){
    segs[rt].cover = segs[rt*2].cover + segs[rt*2+1].cover;
}
void buildtree(int rt,int L,int R){
    if(L == R){
        segs[rt].cover = 0;
        return;
    }
    buildtree(lson);
    buildtree(rson);
    PushUp(rt);
}
void Update(int rt,int L,int R,int id){
    if(L == R){
        segs[rt].cover++;
        return ;
    }
    if(id <= mid){
        Update(lson,id);
    }else{
        Update(rson,id);
    }
    PushUp(rt);
}
int query(int rt,int L, int R,int k){
    if(L == R){
        return L;
    }
    if(k <= segs[rt*2].cover){
        return query(lson,k);
    }else{
        return query(rson,k-segs[rt*2].cover);
    }
}
int main(){
    int n, k, m;
    while(scanf("%d%d%d",&n,&k,&m)!=EOF){
        buildtree(1,1,n);
        LL a;
        for(int i = 1; i <= n; i++){
            scanf("%lld",&a);
            presum[i] = presum[i-1] + a;
        }
        int l, r;
        for(int i = 1; i <= m; i++){
            scanf("%d%d",&l,&r);
            intervals[i].l = l;
            intervals[i].r = r;
        }
        sort(intervals+1,intervals+1+m,cmp);
        for(int i = m-k+1; i <= m; i++){
            Update(1,1,n,intervals[i].l);
        }
        LL ans = 0;
        for(int i = m-k+1; i >= 1; i--){
            int l = query(1,1,n,k);
            if(l <= intervals[i].r){
                ans = max(ans, presum[intervals[i].r] - presum[l-1]);
            }
            Update(1,1,n,intervals[i-1].l);
        }
        printf("%lld\n",ans);
    }
    return 0;
}

/*
5 1 1
1 2 3 4 6
4 5

3 4

*/

  

时间: 2024-11-05 16:39:42

HDU 5700——区间交——————【线段树+枚举】的相关文章

hdu 5700区间交(线段树)

区间交 Time Limit: 8000/4000 MS (Java/Others)    Memory Limit: 65536/65536 K (Java/Others)Total Submission(s): 849    Accepted Submission(s): 377 Problem Description 小A有一个含有n个非负整数的数列与m个区间.每个区间可以表示为li,ri. 它想选择其中k个区间, 使得这些区间的交的那些位置所对应的数的和最大. 例如样例中,选择[2,5]

HDU 5700 区间交 线段树暴力

枚举左端点,然后在线段树内,更新所有左边界小于当前点的区间的右端点,然后查线段树二分查第k大就好 #include <cstdio> #include <cstring> #include <algorithm> using namespace std; typedef long long LL; const int N = 100005; LL a[N]; struct Node{ int l,r; bool operator<(const Node &

HDU 5700 区间交(线段树)

题目链接 区间交 1 #include <bits/stdc++.h> 2 3 using namespace std; 4 5 #define rep(i, a, b) for (int i(a); i <= (b); ++i) 6 #define dec(i, a, b) for (int i(a); i >= (b); --i) 7 8 #define lson i << 1, L, mid 9 #define rson i << 1 | 1, mid

HDU 1823 二维线段树(区间max)

Luck and Love Time Limit: 10000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others)Total Submission(s): 5262    Accepted Submission(s): 1317 Problem Description 世界上上最远的距离不是相隔天涯海角而是我在你面前可你却不知道我爱你                ―― 张小娴 前段日子,枫冰叶子给Wiskey做了

hdu 2665 可持久化线段树求区间第K大值(函数式线段树||主席树)

http://acm.hdu.edu.cn/showproblem.php?pid=2665 Problem Description Give you a sequence and ask you the kth big number of a inteval. Input The first line is the number of the test cases. For each test case, the first line contain two integer n and m (

HDU 3074-Multiply game(线段树:单点更新,区间求积)

Multiply game Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others) Total Submission(s): 1450    Accepted Submission(s): 508 Problem Description Tired of playing computer games, alpc23 is planning to play a game on numbe

HDU 4902 Nice boat(线段树)

HDU Nice boat 题目链接 题意:给定一个序列,两种操作,把一段变成x,把一段每个数字,如果他大于x,就变成他和x的gcd,求变换完后,最后的序列. 思路:线段树,每个结点多一个cover表示该位置以下区间是否数字全相同,然后每次延迟操作,最后输出的时候单点查询即可 代码: #include <cstdio> #include <cstring> #include <algorithm> using namespace std; const int N = 1

hdu 2795 Billboard(线段树)

Billboard Time Limit: 20000/8000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others) Total Submission(s): 10890    Accepted Submission(s): 4827 Problem Description At the entrance to the university, there is a huge rectangular billboard of

HDU 3954 Level up 线段树

---NotOnlySuccess 出的题--- 看了题之后觉得和HDU 4027有点像,给的K很小,只有10,目测只要有人升级的时候直接更新到叶子节点就ok了.不过不同于HDU 4027 的是,那题每一次更新都相当于这题的一次升级操作,这题里面可能会出现一次操作之后没有升级和出现升级两种情况,一时半会没了思路. 无奈去搜题解,发现我只要维护一个区间当中距离升级最近的人所需要的基础升级经验,即不算等级加成的裸的升级经验,如果在一次涨经验之后,出现当前区间当中有人会升级,直接将每一个要升级的人更新