区间交

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. 冲刺NO.10

    Alpha冲刺第十天 站立式会议 项目进展 项目核心功能逐步构建完成,测试工作也已开始.主要对部分功能组合进行测试以测试系统可用性. 问题困难 项目的主要困难在这个时间点主要存在于测试工作中,测试工作 ...

  2. Beta冲刺Day5

    项目进展 李明皇 今天解决的进度 服务器端还未完善,所以无法进行联动调试.对页面样式和逻辑进行优化 明天安排 前后端联动调试 林翔 今天解决的进度 完成维护登录态,实现图片上传,微信开发工具上传图片不 ...

  3. Django 分类标签查找

    from django.conf.urls import url from django.contrib import admin from blog.views import index,stude ...

  4. Python习题(第一课)

    想了想其他的太简单了,还是不放了,剩三题吧. 一.完美立方 编写一个程序,对任给的正整数N (N≤100),寻找所有的四元组(a, b, c, d),使得a^3= b^3 + c^3 + d^3,其中 ...

  5. JAVAEE——BOS物流项目09:业务受理需求分析、创建表、实现自动分单、数据表格编辑功能使用方法和工作单快速录入

    1 学习计划 1.业务受理需求分析 n 业务通知单 n 工单 n 工作单 2.创建业务受理环节的数据表 n 业务通知单 n 工单 n 工作单 3.实现业务受理自动分单 n 在CRM服务端扩展方法根据手 ...

  6. 剑指offer-删除链表中重复的节点

    题目描述   在一个排序的链表中,存在重复的结点,请删除该链表中重复的结点,重复的结点不保留,返回链表头指针. 例如,链表1->2->3->3->4->4->5 处 ...

  7. 用‘+=’拼接字符串,打印时总会出现一个undefined

    var str; for(var i = 0; i < 5; i++){ str += String(i); } console.log(str); 他喵的,打印的结果竟然是"unde ...

  8. 《深入实践Spring Boot》阅读笔记之三:核心技术源代码分析

    刚关注的朋友,可以回顾前两篇文章: 基础应用开发 分布式应用开发 上篇文章总结了<深入实践Spring Boot>的第二部分,本篇文章总结第三部分,也是最后一部分.这部分主要讲解核心技术的 ...

  9. kubernetes进阶(02)kubernetes的node

    一.Node概念 Node是Pod真正运行的主机,可以物理机,也可以是虚拟机. 为了管理Pod,每个Node节点上至少要运行container runtime(比如docker或者rkt). kube ...

  10. HTTP协议扫盲(一)HTTP协议的基本概念和通讯原理

    一.HTTP协议的概念 1.引子  - 从url开始 URL(Uniform Resource Locator) 地址用于描述一个网络上的资源, 基本格式如下 schema://host[:port# ...