区间交

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]与[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
/*
hdu 5700区间交(线段树) problem:
给你一串数字以及m个区间,然后选择其中的k个区间使区间相交区域的和最大 solve:
最开始想的是二分答案然后判断能否找出k个区间使其的和达到,但是推着推着发现和以前的做过的线段树处理区间问题很像
枚举区间的右端点,然后找出左边哪个端点使其刚好覆盖的k个区间,然后求值取最大值
所以需要维护左端的个数,以及查找出第k小的左端点,用线段树解决.
因为枚举的是以当前点为最终区间的右端点,所以出现过的区间要除去(即删除它的左端点). hhh-2016-08-13 15:15:56
*/
#include <iostream>
#include <vector>
#include <cstring>
#include <string>
#include <cstdio>
#include <queue>
#include <algorithm>
#include <functional>
#include <map>
#include <set>
using namespace std;
#define lson (i<<1)
#define rson ((i<<1)|1)
typedef long long ll;
typedef unsigned int ul;
const int INF = 0x3f3f3f3f;
const int maxn = 100000+10;
const int mod = 1e9+7; struct Tree
{
int l,r;
int lval,mid;
} tree[maxn<<2]; void push_up(int i)
{
tree[i].lval = tree[lson].lval+tree[rson].lval;
} void build(int i,int l,int r)
{
tree[i].l = l,tree[i].r = r,tree[i].lval = 0;
if(l == r)
return;
tree[i].mid= (l+r) >>1;
int mid = tree[i].mid;
build(lson,l,mid);
build(rson,mid+1,r);
push_up(i);
} void Insert(int i,int k,int val)
{
if(tree[i].l == tree[i].r && tree[i].l == k)
{
tree[i].lval += val;
// cout <<"pos: " << tree[i].l <<" val: "<<tree[i].lval<< endl;
return ;
}
// cout <<tree[i].l << " "<< tree[i].r << "k "<<k <<" val: "<<val<< endl;
int mid = tree[i].mid; if(k <= mid)
Insert(lson,k,val);
else
Insert(rson,k,val);
push_up(i);
// cout <<tree[i].l << " "<< tree[i].r <<" val: "<<tree[i].lval<< endl;
} int query(int i ,int k)
{
// cout << tree[i].lval << " " <<tree[i].l << " "<< tree[i].r <<endl;
if(tree[i].l == tree[i].r)
return tree[i].l;
if(k <= tree[lson].lval)
return query(lson,k);
else
return query(rson,k-tree[lson].lval);
push_up(i);
} int query_num(int i,int l,int r)
{
if(tree[i].l >= l && tree[i].r <= r)
{
return tree[i].lval;
}
int num = 0;
if(l <= tree[i].mid)
num += query_num(lson,l,r);
if(r > tree[i].mid)
num += query_num(rson,l,r);
return num;
} struct node
{
int l,r;
} pnode[maxn]; ll Max(ll a,ll b)
{
if(a < b)
return b;
return a;
} bool cmp(node a,node b)
{
if(a.r != b.r)
return a.r < b.r;
else
return a.l < b.l;
}
ll num[maxn];
int main()
{
int n,k,m;
while(scanf("%d%d%d",&n,&k,&m) != EOF)
{
num[0] = 0;
build(1,1,n);
for(int i = 1; i <= n; i++)
{
scanf("%I64d",&num[i]);
num[i] += num[i-1];
}
for(int i = 1; i <= m; i++)
{
scanf("%d%d",&pnode[i].l,&pnode[i].r);
Insert(1,pnode[i].l,1);
}
sort(pnode+1,pnode+m+1,cmp); int now = 1;
ll ans = 0;
for(int i = 1; i <= n && now <= m; i++)
{
if(i == pnode[now].r)
{
// cout << "r:"<<pnode[now].r << endl;
int t = query_num(1,1,i);
if(t >= k)
{
int l = query(1,k);
// cout<<"l:" << l << endl;
ans = Max(ans,num[i]-num[l-1]);
}
// cout<<"ans:"<<ans <<" t:" << t << endl;
}
while(now <= m && pnode[now].r == i)
{
Insert(1,pnode[now].l,-1);
now ++;
}
}
cout << ans <<endl;
}
return 0;
} /*
5 2 3
1 2 3 4 5
1 2
3 4
5 5
*/

  

hdu 5700区间交(线段树)的更多相关文章

  1. HDU 5700 区间交 线段树暴力

    枚举左端点,然后在线段树内,更新所有左边界小于当前点的区间的右端点,然后查线段树二分查第k大就好 #include <cstdio> #include <cstring> #i ...

  2. HDU 5700 区间交 离线线段树

    区间交 题目连接: http://acm.hdu.edu.cn/showproblem.php?pid=5700 Description 小A有一个含有n个非负整数的数列与m个区间.每个区间可以表示为 ...

  3. HDU 5700 区间交(线段树)

    [题目链接] http://acm.hdu.edu.cn/showproblem.php?pid=5700 [题目大意] 给出一个长度为n的数列和m个区间,现在求k个区间,使得他们的区间交内的数列项和 ...

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

    区间交 Time Limit: 8000/4000 MS (Java/Others)    Memory Limit: 65536/65536 K (Java/Others)Total Submiss ...

  5. HDU 5700 区间交

    枚举起点 二分终点 树状数组check #include<iostream> #include<cstring> #include<cmath> #include& ...

  6. HDU 1540 区间合并线段树

    题目大意: 就是给定一堆位置,进行删除还原,最后找到 t 位置上的最大连续位置 #include <cstdio> #include <cstring> #include &l ...

  7. Snacks HDU 5692 dfs序列+线段树

    Snacks HDU 5692 dfs序列+线段树 题意 百度科技园内有n个零食机,零食机之间通过n−1条路相互连通.每个零食机都有一个值v,表示为小度熊提供零食的价值. 由于零食被频繁的消耗和补充, ...

  8. 【BZOJ4653】【NOI2016】区间(线段树)

    [BZOJ4653][NOI2016]区间(线段树) 题面 BZOJ 题解 \(NOI\)良心送分题?? 既然是最大长度减去最小长度 莫名想到那道反复减边求最小生成树 从而求出最小的比值 所以这题的套 ...

  9. BZOJ_4653_[Noi2016]区间_线段树+离散化+双指针

    BZOJ_4653_[Noi2016]区间_线段树+离散化+双指针 Description 在数轴上有 n个闭区间 [l1,r1],[l2,r2],...,[ln,rn].现在要从中选出 m 个区间, ...

随机推荐

  1. Android网络传输中必用的两个加密算法:MD5 和 RSA 及Base64加密总结

    (1)commons-codec包简介 包含一些通用的编码解码算法.包括一些语音编码器,Hex,Base64.MD5 一.md5.base64.commons-codec包 commons-codec ...

  2. 201421123042 《Java程序设计》第10周学习总结

    1. 本周学习总结 1.1 以你喜欢的方式(思维导图或其他)归纳总结异常相关内容. 2. 书面作业 本次PTA作业题集异常 1. 常用异常 结合题集题目7-1回答 1.1 自己以前编写的代码中经常出现 ...

  3. Scala 对象

    1. 单例对象 对于任何你在Java中会使用单例对象的地方, 在scala中都可以使用对象来实现; scala字段没有静态方法或者静态字段, 可以使用object语法结构达到同样的效果,对象(obje ...

  4. 根据HTML5的新方法 drag & drop 方法实现表格拖拽实例

    上一次学习了html5的drag和drop方法,传送门  就自己写了个例子加深自己对drag和drop的理解.不过一开始不是很简单,遇到了不少问题.还好网络万能的,什么都能查到,总算完成了. 说明和详 ...

  5. JAVA_SE基础——41.instanceof关键字(运算符)

    instanceof 关键字 instanceof关键字的作用:判断一个对象是否属于指定的类别. instanceof关键字的使用前提:判断的对象与指定的类别必须要存在继承或者实现的关系.关于实现以后 ...

  6. Ubuntu Desktop 16.04 LTS 下成功配置Jupyter的两个python内核版本(2.7x,3.5x)

    Ubuntu  Desktop 16.04 LTS 安装好系统默认就有python两个不同版本(2.7.12和3.5.2) 现在来熟悉一下jupyter的对python这两个不同python版本的内核 ...

  7. awk sed tr替换换行符为逗号,并合并为一行

    在群里看到的.记录以备用.  sed 帮助命令:http://man.linuxde.net/sed 文件里有如下行,我想将每行的回车符替换为逗号,并将所有行合并到一行,用awk或sed怎么写啊TOP ...

  8. emqtt 试用(一)安装和测试

    一.安装 http://emqtt.io/docs/v2/getstarted.html http://emqtt.io/docs/v2/advanced.html http://emqtt.io/d ...

  9. Docker学习笔记 - Docker容器内部署redis

    Docker学习笔记(2-4)Docker应用实验-redist server 和client的安装使用 一.获取redis容器(含客户端和服务端) 二.创建服务端容器 1.在终端A中运行redis- ...

  10. 基于python的统计公报关键数据爬取

    # -*- coding: utf-8 -*- """ Created on Wed Nov 8 14:23:14 2017 @author: 123 "&qu ...