题意就是给出多个区间,要求去掉两个区间,使得剩下的区间覆盖范围最大。

当然比赛的时候还是没能做出来,不得不佩服大佬的各种姿势。

当时我想的是用线段树维护区间和,然后用单点判0,维护区间间断个数。然后打到一半,就发现想法有问题。

这道题正解就是简单的前缀和,或者DP。

我为了更加深入理解,两种方法都试了试。

前缀和版本:

由于题目给的范围是5000,明显支持N^2,于是我们枚举去掉的两个,刚好满足,那么要如何才能O(1)的得到答案?

我们其实可以这样,我们知道它的总覆盖的数目,这是非常容易求出的。减去枚举的区间对答案的贡献即可。那么如何减去贡献,我们知道两个区间,可能有两种情况,一种是有交集部分,这一部分需要在交集区间上减去2次,而剩下的部分需要在剩下部分减去1次就行。

那么如果要对答案产生影响,答案就是

总覆盖区间=枚举区间交集部分刚好被覆盖两次+剩下交集为空的区间被覆盖一次的个数

考虑如何求被覆盖一次的个数,通过维护差分,求得每个点的被覆盖次数

枚举区间,就能求得每个区间,被覆盖次数为1的个数(不会有交集)。

那么如何求两个区间交集部分刚好被覆盖两次的值呢???

很简单,由于这个区间交集是不确定的。那么我们如何求呢?

我们已经知道每个点的覆盖次数。

那么我们可以知道刚好被覆盖两次的点,所在位置。

利用前缀和,和差分的思想,我们可以用一个新的前缀和数组,记录当前被覆盖的次数刚好被覆盖次数为1的位置。

然后求前缀和,这样我们就能快速的求出,[L,R]区间内,刚好为2的个数。

这样答案也出来了,注意枚举区间可能交集可能不存在,这样就不用算覆盖次数为2的个数。

#include<iostream>
#include<stdio.h>
#include<string.h>
#include<algorithm>
#include<math.h>
using namespace std;
const int maxx = ;
int pre[][maxx];
struct node{
int l,r;
}a[maxx];
int main(){
int n,q,sum,ans;
while(~scanf("%d%d",&n,&q)){
ans=;
sum=;
memset(pre,,sizeof(pre));
for (int i=;i<=q;i++){
scanf("%d%d",&a[i].l,&a[i].r);
pre[][a[i].l]+=;
pre[][a[i].r+]-=;
}
for (int i=;i<=n;i++){
pre[][i]+=pre[][i-];//每个位置的情况
if (pre[][i])sum++;
}
for (int i=;i<=n;i++){
if (pre[][i]==)
pre[][i]++;//每个位置个数是否是2
}
for (int i=;i<=n;i++){
pre[][i]+=pre[][i-];//次数==2的前i项个数
}
for (int i=;i<=q;i++){
for (int j=a[i].l;j<=a[i].r;j++){
if (pre[][j]==){
pre[][i]++;//在第i个区间内部有多少个数为1的个数
}
}
}
for (int i=;i<=q;i++){
for (int j=i+;j<=q;j++){
int l=max(a[i].l,a[j].l);
int r=min(a[i].r,a[j].r);
int tmp=sum;
if (r>=l)
tmp-=(pre[][r]-pre[][l-]);
tmp-=(pre[][j]+pre[][i]);
ans=max(ans,tmp);
}
}
printf("%d\n",ans);
}
return ;
}

那么我们考虑DP做法。

这个DP和我前几天做的DP非常像。

我们可以构建这样DP

DP[i]代表,选到i位置,选择k个的最优解。用L[i]表示某个覆盖到i位置的区间,往左能覆盖的最小坐标

那么转移方程就是:

dp[j]=max(dp[L[j]-1]+j-L[j]+1,dp[j])

同时要是短的段落,能覆盖的个数是大于当前的,可以更新更长的段。

#include<iostream>
#include<stdio.h>
#include<string.h>
#include<algorithm>
#define rep(i,j,k) for(int i=j;i<=k;i++)
#define per(i,j,k) for(int i=j;i>=k;i--)
using namespace std;
const int maxx = +;
int dp[maxx];
int L[maxx];
int main()
{
int n,q,tmp,l,r;
while(~scanf("%d%d",&n,&q))
{
rep(i,,n)
{
L[i]=i+;
dp[i]=;
}
rep(i,,q)
{
scanf("%d%d",&l,&r);
rep(j,l,r)
{
L[j]=min(L[j],l);
}
}
// for (int i=1;i<=n;i++){
// printf("%d ",L[i]);
// }
// cout<<endl;
rep(i,,q-)
{
per(j,n,)
{
dp[j]=max(dp[L[j]-]+j-L[j]+,dp[j]);
}
rep(j,,n){
dp[j]=max(dp[j],dp[j-]);
}
}
printf("%d\n",dp[n]);
}
return ;
}

Educational Codeforces Round 61 (Rated for Div. 2)-C. Painting the Fence 前缀和优化的更多相关文章

  1. Educational Codeforces Round 73 (Rated for Div. 2) D. Make The Fence Great Again(DP)

    链接: https://codeforces.com/contest/1221/problem/D 题意: You have a fence consisting of n vertical boar ...

  2. Educational Codeforces Round 61 (Rated for Div. 2) D,F题解

    D. Stressful Training 题目链接:https://codeforces.com/contest/1132/problem/D 题意: 有n台电脑,每台电脑都有初始电量ai,也有一个 ...

  3. Educational Codeforces Round 61 (Rated for Div. 2) E 多重背包优化

    https://codeforces.com/contest/1132/problem/E 题意 有8种物品,重量是1~8,每种数量是\(cnt[i]\)(1e16),问容量为W(1e18)的背包最多 ...

  4. Educational Codeforces Round 61 (Rated for Div. 2)

    A. Regular Bracket Sequence 题意:给出四种括号的数量 ((  )) ()  )( 问是否可以组成合法的序列(只能排序不能插在另外一个的中间) 思路: 条件一:一个或 n个) ...

  5. Educational Codeforces Round 61 (Rated for Div. 2) E. Knapsack

    非常经典的dp题,因为1至8的最大公约数是840,任何一个数的和中840的倍数都是可以放在一起算的, 所以我只需要统计840*8的值(每个数字(1-8)的sum%840的总和),剩下都是840的倍数 ...

  6. Educational Codeforces Round 61 (Rated for Div. 2) G(线段树,单调栈)

    #include<bits/stdc++.h>using namespace std;int st[1000007];int top;int s[1000007],t[1000007];i ...

  7. Educational Codeforces Round 61 (Rated for Div. 2)F(区间DP,思维,枚举)

    #include<bits/stdc++.h>typedef long long ll;const int inf=0x3f3f3f3f;using namespace std;char ...

  8. Educational Codeforces Round 61 (Rated for Div. 2)D(二分,模拟,思维)

    #include<bits/stdc++.h>using namespace std;typedef long long ll;int n,k;ll a[200007],b[200007] ...

  9. Educational Codeforces Round 55 (Rated for Div. 2) C. Multi-Subject Competition 【vector 预处理优化】

    传送门:http://codeforces.com/contest/1082/problem/C C. Multi-Subject Competition time limit per test 2 ...

随机推荐

  1. spring设计模式_代理模式

    代理模式应该是Spring核心设计模式之一了 先说下代理模式特性: 1.有代理人和被代理人 2.对于被代理的人来说,这件事情是一定要做的,但是我又不想做,所有就找代理人来做. 3.需要获取到被代理人的 ...

  2. 【工利其器】必会工具之(二)Android开发者官网篇

    前言 当刚开始踏入Android程序员这个行业的时候,想必绝大多数的人都和笔者一样,热血沸腾,激情四射,买了很多讲解Android开发的书籍.当开发某个功能需要学习某方面知识的时候,大家又成了“面向百 ...

  3. springboot~mockMvc和asciidoctor生成基于TDD的API文档

    API文档是前端与后端快速开发,减少沟通成本的必要条件,有一份完善的文档是很必要的,由通过测试来生成文档的好处就是:测试数据有了,测试返回结果有了,而且可以对这些字段进行说明,很清晰,在springb ...

  4. Windows Server 2012 R2安装SqlServer 2016

    1.系统安装 微软操作系统 Windows Server 2012 R2 官方原版镜像 Windows Server 2012 R2 是由微软公司(Microsoft)设计开发的新一代的服务器专属操作 ...

  5. Spring Boot配置拦截器及实现跨域访问

    拦截器功能强大,能够深入方法前后,常应用于日志记录.权限检查和性能检测等,几乎是项目中不可或缺的一部分,本文就来实现Spring Boot自定义拦截器的配置. 理论指导 问:Spring Boot怎么 ...

  6. javascript小记三则:ASP.NET启动web调试,窗体自动放大的方法

    源码如下,简单一句JS,轻松解决窗体不会自动放大的浏览器: <!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN& ...

  7. VUE v-for问题

    今天写一个拖动然后使装备交换的功能,在背包格子里 发现直接设置Bags数组的项,v-for渲染出来的列表不会对应改变,只有设置值才会改变 有点拗口,贴代码吧 var repear = this.Bag ...

  8. 80后程序员降薪6K,预感中年危机来袭,准备跳槽却碰壁

    一提及程序员,很多人想到的都是“工资高”“技术好”诸如此类的,可见程序员是个非常赚钱的职业,所以每年都会有很多毕业生来选择这个行业. 但是社会是公平的,不要只看程序员表面上的光鲜亮丽,其背后也有很多的 ...

  9. JVM虚拟机基本概念

    一.JVM运行时数据区域1.1.程序计数器 一块较小的内存空间,当前线程所执行的字节码指示器.每个线程有一个独立的程序计数器1.2.Java虚拟机栈 线程私有,生命周期与线程相同 每个方法在执行时会创 ...

  10. 编程心法 之 Scrum - Agile 敏捷开发

    Scrum是一种敏捷开发的方法 先定一个能达到的小目标 Scrum 团队 包括产品负责人.开发团队和Scrum Master Product Owner 产品负责人:管理代办事项和优先级的唯一负责人. ...