【bzoj3747】[POI2015]Kinoman 线段树区间合并

时间:2022-08-28 18:40:48

题目描述

一个长度为n的序列,每个数为1~m之一。求一段连续子序列,使得其中之出现过一次的数对应的价值之和最大。

输入

第一行两个整数n,m(1<=m<=n<=1000000)。
第二行包含n个整数f[1],f[2],…,f[n](1<=f[i]<=m)。
第三行包含m个整数w[1],w[2],…,w[m](1<=w[j]<=1000000)。

输出

输出观看且仅观看过一次的电影的好看值的总和的最大值。

样例输入

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

样例输出

15


题解

线段树区间合并

考虑从左到右添加数的过程:选出的以该数为右端点的连续子序列中,该数的贡献为w[a[i]],该数上一个相同的数的贡献为-w[a[i]](因为选择了该数就会重复,需要把该数的贡献减掉),再前面相同的数的贡献为0。

于是可以使用线段树实现单点修改和查询最大连续子段和(实际上是1~i的最大包含右端点连续子段和,但是考虑到如果最大连续子段和不包含右端点,那么以前它的贡献不会更差,因此无需讨论以前的贡献)。

时间复杂度为$O(n\log n)$。由于空间原因,数组不能无脑开到4倍,需要开$3.5*10^6$可过。

#include <cstdio>
#include <algorithm>
#define N 1000010
#define M 3500010
using namespace std;
int pos[N] , last[N] , v[N] , w[N];
long long sum[M] , lv[M] , rv[M] , tv[M];
inline void pushup(int x)
{
int l = x << 1 , r = x << 1 | 1;
sum[x] = sum[l] + sum[r];
lv[x] = max(lv[l] , sum[l] + lv[r]);
rv[x] = max(rv[r] , sum[r] + rv[l]);
tv[x] = max(rv[l] + lv[r] , max(tv[l] , tv[r]));
}
void update(int p , int a , int l , int r , int x)
{
if(l == r)
{
sum[x] = a , lv[x] = rv[x] = tv[x] = max(a , 0);
return;
}
int mid = (l + r) >> 1;
if(p <= mid) update(p , a , l , mid , x << 1);
else update(p , a , mid + 1 , r , x << 1 | 1);
pushup(x);
}
int main()
{
int n , m , i;
long long ans = 0;
scanf("%d%d" , &n , &m);
for(i = 1 ; i <= n ; i ++ ) scanf("%d" , &v[i]);
for(i = 1 ; i <= m ; i ++ ) scanf("%d" , &w[i]);
for(i = 1 ; i <= n ; i ++ )
{
last[i] = pos[v[i]] , pos[v[i]] = i;
update(i , w[v[i]] , 1 , n , 1);
if(last[i])
{
update(last[i] , -w[v[i]] , 1 , n , 1);
if(last[last[i]]) update(last[last[i]] , 0 , 1 , n , 1);
}
ans = max(ans , tv[1]);
}
printf("%lld\n" , ans);
return 0;
}

【bzoj3747】[POI2015]Kinoman 线段树区间合并的更多相关文章

  1. BZOJ3747&colon;&lbrack;POI2015&rsqb;Kinoman&lpar;线段树&rpar;

    Description 共有m部电影,编号为1~m,第i部电影的好看值为w[i]. 在n天之中(从1~n编号)每天会放映一部电影,第i天放映的是第f[i]部. 你可以选择l,r(1<=l< ...

  2. POJ 3667 Hotel&lpar;线段树 区间合并&rpar;

    Hotel 转载自:http://www.cnblogs.com/scau20110726/archive/2013/05/07/3065418.html [题目链接]Hotel [题目类型]线段树 ...

  3. HDU 3911 线段树区间合并、异或取反操作

    题目:http://acm.hdu.edu.cn/showproblem.php?pid=3911 线段树区间合并的题目,解释一下代码中声明数组的作用: m1是区间内连续1的最长长度,m0是区间内连续 ...

  4. HDU 3911 Black And White(线段树区间合并&plus;lazy操作)

    开始以为是水题,结果...... 给你一些只有两种颜色的石头,0为白色,1为黑色. 然后两个操作: 1 l r 将[ l , r ]内的颜色取反 0 l r 计算[ l , r ]内最长连续黑色石头的 ...

  5. HYSBZ 1858 线段树 区间合并

    //Accepted 14560 KB 1532 ms //线段树 区间合并 /* 0 a b 把[a, b]区间内的所有数全变成0 1 a b 把[a, b]区间内的所有数全变成1 2 a b 把[ ...

  6. poj3667 线段树 区间合并

    //Accepted 3728 KB 1079 ms //线段树 区间合并 #include <cstdio> #include <cstring> #include < ...

  7. hdu3911 线段树 区间合并

    //Accepted 3911 750MS 9872K //线段树 区间合并 #include <cstdio> #include <cstring> #include &lt ...

  8. 线段树&lpar;区间合并&rpar; POJ 3667 Hotel

    题目传送门 /* 题意:输入 1 a:询问是不是有连续长度为a的空房间,有的话住进最左边 输入 2 a b:将[a,a+b-1]的房间清空 线段树(区间合并):lsum[]统计从左端点起最长连续空房间 ...

  9. HDU 3308 LCIS &lpar;线段树区间合并&rpar;

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=3308 题目很好懂,就是单点更新,然后求区间的最长上升子序列. 线段树区间合并问题,注意合并的条件是a[ ...

随机推荐

  1. Just a Hook&lpar;HDU1698 线段树的简单应用&rpar;

    Just a Hook Time Limit: 4000/2000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others) Prob ...

  2. Grunt参考

    Grunt参考: http://www.cnblogs.com/yexiaochai/p/3603389.html http://blog.csdn.net/wangfupeng1988/articl ...

  3. &lbrack;编织消息框架&rsqb;&lbrack;网络IO模型&rsqb;Netty Reactor

    严格来讲Netty Reactor是一种设计模式,一听模式两字就知道了吧,套路哈哈 Reactor中文译为“反应堆”. 看图netty处理流程 1.netty server 至少有两组reactor. ...

  4. java中lamda表达式的应用

    lamda表达式主要是为了解决匿名内部类的繁琐过程 范例:简单的lamda表达式 此处使用匿名内部类 package com.java.demo; interface IMessage{ public ...

  5. LeetCode(29)-Plus One

    题目: Given a non-negative number represented as an array of digits, plus one to the number. The digit ...

  6. Oracle体系结构之密码文件管理

    oracle密码文件主要用来控制sysdba和sysoper用户用于远程登录.通常,oracle用户登录database有两种方式,一种是通过本地操作系统验证登录,一种是通过密码文件验证登录. 操作系 ...

  7. BZOJ 3731&colon; Gty的超级妹子树

    3731: Gty的超级妹子树 Time Limit: 7 Sec  Memory Limit: 32 MBSubmit: 346  Solved: 96[Submit][Status][Discus ...

  8. C语言——第一章,程序设计和C语言

    第一章,程序设计和C语言 一,程序和程序语言 程序:完成某项事物所预设的活动方式和活动过程 程序设计:人们描述(指程序)计算机要做的工作 程序设计语言及发展 1,机器语言    2,汇编语言    3 ...

  9. Unix系统中system函数的返回值

    网上关于system函数的返回值说明很多很详细但却不直观,这里搬出apue 3rd Editon中实现system函数的代码来说明其返回值. #include <sys/wait.h> # ...

  10. ElasticSearch搜索服务技术

    ElasticSearch 基于的lucene开发的搜索服务技术;天生支持分布式; Es的结构 gatway:存储层,所有的数据可以存储在本地(多个es节点形成分布式存储),hdfs输出位置,共享文件 ...