●CodeForces 280D k-Maximum Subsequence Sum
题链:
http://codeforces.com/problemset/problem/280/D
题解:
神题,巨恶心。
(把原来的那个dp题升级为:序列带修 + 多次询问区间[l,r]内取不超过k段的不重叠子串,使得其和最大)。
按费用流的思路来看,建图方法如下:
每个点拆成两个点 i , i' ,建立超源 S和超汇 T
i -> i' : (1,a[i])
S -> i : (1,0)
i'-> T : (1,0)
i'-> i+1: (1,0)
那么对于某段区间,按照spfa最长路费用流去一条路一条路增广,
直到某个时候增广数==k或者增广路的费用为负数就停止。
分析其增广方式,不难发现一个重要特点(可以自己简单伪证一下哈):
每次找到增广路都是连续的一段,即对应着序列区间上和最大的连续的一段。
接下来增广操作,就会取出这一段的权值和,并把这一段的所有数全部 * -1。(就是增广后的反向边的花费)
所以就用线段树维护区间最大子段和以及最小子段和,
并且要支持单点修改和区间 * -1操作。
每次就取出[l,r]区间内的最大的子段,并把对应的子段全部 * -1,
如果取得次数==k或者最大的子段和为负数就停止。
。。。这个线段树不是一般的烦。
代码:
#include<cstdio>
#include<cstring>
#include<iostream>
#define MAXN 100005
#define INF 0x3f3f3f3f
using namespace std;
int a[MAXN];
struct data{
int sval,lval,rval,sum,sl,sr,lr,rl;
void init(bool type){
if(type) lval=rval=sval=-INF;
else lval=rval=sval=+INF;
}
void reverse(){
sval*=-1; lval*=-1; rval*=-1; sum*=-1;
}
void update(const data &l,const data &r,bool type){
sum=l.sum+r.sum;
if(type)
lval=max(l.lval,l.sum+r.lval), rval=max(r.rval,r.sum+l.rval),
sval=max(max(l.sval,r.sval),l.rval+r.lval);
else
lval=min(l.lval,l.sum+r.lval), rval=min(r.rval,r.sum+l.rval),
sval=min(min(l.sval,r.sval),l.rval+r.lval); if(l.lval==lval) lr=l.lr; else lr=r.lr; if(r.rval==rval) rl=r.rl; else rl=l.rl; if(l.sval==sval) sl=l.sl,sr=l.sr;
else if(r.sval==sval) sl=r.sl,sr=r.sr;
else sl=l.rl,sr=r.lr;
}
};
struct info{
data maxi,mini;
void init(){
maxi.init(1);
mini.init(0);
}
};
struct SGT{
#define ls lson[u]
#define rs rson[u]
bool lazy[MAXN<<1];
int lson[MAXN<<1],rson[MAXN<<1],rt,sz;
info node[MAXN<<1];
void init(){
rt=sz=0;
memset(lazy,0,sizeof(lazy));
memset(lson,0,sizeof(lson));
memset(rson,0,sizeof(rson));
for(int i=0;i<(MAXN<<1);i++)
node[i].init();
}
void pushup(info &now,const info &l,const info &r){
now.maxi.update(l.maxi,r.maxi,1);//___________________维护最大连续和_1__
now.mini.update(l.mini,r.mini,0);//___________________维护最小连续和_0__
}
void pushdown(int u){
node[ls].maxi.reverse(); node[ls].mini.reverse();
node[rs].maxi.reverse(); node[rs].mini.reverse();
swap(node[ls].maxi,node[ls].mini);
swap(node[rs].maxi,node[rs].mini);
lazy[u]^=1; lazy[ls]^=1; lazy[rs]^=1;
}
void build(int &u,int l,int r){
u=++sz;
if(l==r) {
node[u].maxi=(data){a[l],a[l],a[l],a[l],l,r,r,l};
node[u].mini=(data){a[l],a[l],a[l],a[l],l,r,r,l};
return;
}
int mid=(l+r)>>1;
build(ls,l,mid);
build(rs,mid+1,r);
pushup(node[u],node[ls],node[rs]);
}
void modify(int u,int l,int r,int p){
if(l==r){
node[u].maxi=(data){a[l],a[l],a[l],a[l],l,r,r,l};
node[u].mini=(data){a[l],a[l],a[l],a[l],l,r,r,l};
return;
}
if(lazy[u]) pushdown(u);
int mid=(l+r)>>1;
if(p<=mid) modify(ls,l,mid,p);
else modify(rs,mid+1,r,p);
pushup(node[u],node[ls],node[rs]);
}
void modify(int u,int l,int r,int al,int ar){
if(al<=l&&r<=ar){
node[u].maxi.reverse(); node[u].mini.reverse();
swap(node[u].maxi,node[u].mini);
lazy[u]^=1; return;
}
if(lazy[u]) pushdown(u);
int mid=(l+r)>>1;
if(al<=mid) modify(ls,l,mid,al,ar);
if(mid<ar) modify(rs,mid+1,r,al,ar);
pushup(node[u],node[ls],node[rs]);
}
info query(int u,int l,int r,int al,int ar){
if(al<=l&&r<=ar) return node[u];
info now,lnode,rnode;
now.init(); lnode.init(); rnode.init();
if(lazy[u]) pushdown(u);
int mid=(l+r)>>1;
if(al<=mid) lnode=query(ls,l,mid,al,ar);
if(mid<ar) rnode=query(rs,mid+1,r,al,ar); if(mid<al) now=rnode;
else if(ar<=mid) now=lnode;
else pushup(now,lnode,rnode);
return now;
}
#undef ls
#undef rs
}T1;
int N,M,ans;
void dfs(int k,int l,int r){
info now=T1.query(T1.rt,1,N,l,r);
if(now.maxi.sval<=0) return;
ans+=now.maxi.sval;
T1.modify(T1.rt,1,N,now.maxi.sl,now.maxi.sr);
if(k-1)dfs(k-1,l,r);
T1.modify(T1.rt,1,N,now.maxi.sl,now.maxi.sr);
}
int main()
{
//freopen("280D.in","r",stdin);
scanf("%d",&N); T1.init();
for(int i=1;i<=N;i++) scanf("%d",&a[i]);
T1.build(T1.rt,1,N);
scanf("%d",&M);
for(int i=1,c,l,r,k;i<=M;i++){
scanf("%d",&c);
if(!c){
scanf("%d",&k); scanf("%d",&a[k]);
T1.modify(T1.rt,1,N,k);
}
else{
scanf("%d%d%d",&l,&r,&k);
ans=0;
dfs(k,l,r);
printf("%d\n",ans);
}
}
return 0;
}
●CodeForces 280D k-Maximum Subsequence Sum的更多相关文章
- 中国大学MOOC-陈越、何钦铭-数据结构-2015秋 01-复杂度2 Maximum Subsequence Sum (25分)
01-复杂度2 Maximum Subsequence Sum (25分) Given a sequence of K integers { N1,N2, ..., NK }. ...
- PAT1007:Maximum Subsequence Sum
1007. Maximum Subsequence Sum (25) 时间限制 400 ms 内存限制 65536 kB 代码长度限制 16000 B 判题程序 Standard 作者 CHEN, Y ...
- PTA (Advanced Level) 1007 Maximum Subsequence Sum
Maximum Subsequence Sum Given a sequence of K integers { N1, N2, ..., NK }. A continuous su ...
- 【DP-最大子串和】PAT1007. Maximum Subsequence Sum
1007. Maximum Subsequence Sum (25) 时间限制 400 ms 内存限制 32000 kB 代码长度限制 16000 B 判题程序 Standard 作者 CHEN, Y ...
- PAT Maximum Subsequence Sum[最大子序列和,简单dp]
1007 Maximum Subsequence Sum (25)(25 分) Given a sequence of K integers { N~1~, N~2~, ..., N~K~ }. A ...
- PAT甲 1007. Maximum Subsequence Sum (25) 2016-09-09 22:56 41人阅读 评论(0) 收藏
1007. Maximum Subsequence Sum (25) 时间限制 400 ms 内存限制 65536 kB 代码长度限制 16000 B 判题程序 Standard 作者 CHEN, Y ...
- PAT 甲级 1007 Maximum Subsequence Sum (25)(25 分)(0不是负数,水题)
1007 Maximum Subsequence Sum (25)(25 分) Given a sequence of K integers { N~1~, N~2~, ..., N~K~ }. A ...
- PAT 1007 Maximum Subsequence Sum(最长子段和)
1007. Maximum Subsequence Sum (25) 时间限制 400 ms 内存限制 65536 kB 代码长度限制 16000 B 判题程序 Standard 作者 CHEN, Y ...
- pat1007. Maximum Subsequence Sum (25)
1007. Maximum Subsequence Sum (25) 时间限制 400 ms 内存限制 65536 kB 代码长度限制 16000 B 判题程序 Standard 作者 CHEN, Y ...
- PTA 01-复杂度2 Maximum Subsequence Sum (25分)
题目地址 https://pta.patest.cn/pta/test/16/exam/4/question/663 5-1 Maximum Subsequence Sum (25分) Given ...
随机推荐
- 201621123043 《Java程序设计》第9周学习总结
1. 本周学习总结 1.1 以你喜欢的方式(思维导图或其他)归纳总结集合与泛型相关内容. 泛型的定义: 泛型,即"参数化类型".一提到参数,最熟悉的就是定义方法时有形参,然后调用此 ...
- Flask 扩展 Mail
安装 pip install flask-mail from flask import Flask from flask_mail import Mail, Message app = Flask(_ ...
- day-3 python多线程编程知识点汇总
python语言以容易入门,适合应用开发,编程简洁,第三方库多等等诸多优点,并吸引广大编程爱好者.但是也存在一个被熟知的性能瓶颈:python解释器引入GIL锁以后,多CPU场景下,也不再是并行方式运 ...
- video与audio的使用
HTML5 DOM 为 <audio> 和 <video> 元素提供了方法.属性和事件. 这些方法.属性和事件允许您使用 JavaScript 来操作 <audio> ...
- JAVA_SE基础——65.StringBuffer类 ②
字符串特点:字符串是常量:它们的值在创建之后不能更改. 字符串的内容一旦发生了变化,那么马上会创建一个新 的对象. 注意: 字符串的内容不适宜频繁修改,因为一旦修改马上就会创建一个新的对象 ...
- 工频相位无线同步模块PSYN5000系列在高压设备状态检测和局部放电故障定位的应用方案
关键词: PSYN5000,无线同步模块,工频相位,局部放电,在线监测,高压设备,设备状态,故障定位. 前言: 在电力监测领域,出于方便和安全考虑,有些系统不得不采用无线通信的方式,在这样一个无线通信 ...
- webpack你值得拥有-从四个核心配置谈起
很久没有发文章了,但是强调一点,大-熊同学最近可没闲着.学习算法,复习计算机网络,也顺便学习了一下webpack,看了看操作系统(没办法,都没学,要是不学连实习笔试都过不了,伤心--).本来比较纠结是 ...
- SSO的全方位解决方案 - Kerberos协议(RFC 1510)
一.桌面SSO和WEB-SSO的局限性 前面我们的解决方案(桌面SSO和WEB-SSO)都有一个共性:要想将一个应用集成到我们的SSO解决方案中,或多或少的需要修改应用程序. Web应用需要配置一个我 ...
- Bootstrap 做一个简单的母版页
随便搭的一个母版页,不太好看,只是为了看效果....请勿吐槽. 效果如图: 一.新建母版页,引入Bootstrap相关js文件 <link href="../css/bootstrap ...
- 深入理解JavaScript的this指向问题
Javascript的this用法 this是Javascript语言的一个关键字.它代表函数运行时,自动生成的一个内部对象,只能在函数内部使用.比如: function test(){ this.x ...