noip模拟测试31
终于有时间写博客了,前面一直咕咕咕都快变成一只公鸡了......这次考试,真的很意外,我在考场上觉得自己打出了T1的正解,样例一拍就过,还跑得嘎嘎快,然后T2,T3码了两个暴力,觉得自己应该能100pts+,结果竟然....爆蛋了,T1思路出现了问题,T2打假了,T3TLE飞起,但是收获还是有的,以后注意多加思考,看看自己的思路有没有正确性,还有,一定要用暴力程序进行验证(暴力不要打假...),样例真是太水了....
T1 Game
思路:最大得分可以利用线段树很容易求出,如果没有字典序最大的限制,那么这道题就非常简单,但这只是如果,现在这道题有了双重限制,听某巨佬的叙述,得知一般有双重限制的题一般有两种思考方向,一种是二分,另一种是主席树,那么很显然,这道题我们可以考虑二分的思想,具体来说就是我们先通过线段树求出最大得分,然后考虑这样一件事情,如果有一对点可以对答案造成贡献,那么我们将他们同时删去必然会使更新后的最大得分-1,这样我们就可以进行二分查找了,对于可以造成贡献的点,我们二分出他可以对应的最大的另外一个点,具体实现见代码:
AC_code
#include<bits/stdc++.h>
#define re register int
#define ii inline int
#define iv inline void
#define lc (rt<<1)
#define rc (rt<<1|1)
using namespace std;
const int N=200010;
const int INF=1e9+10;
int n;
int a[N],b[N];
multiset<int> sm;
ii read()
{
int x=0;
bool f=1;
char ch=getchar();
while(ch<'0'||ch>'9')
{
if(ch=='-')
f=0;
ch=getchar();
}
while(ch>='0'&&ch<='9')
{
x=(x<<1)+(x<<3)+(ch^48);
ch=getchar();
}
return f?x:(-x);
}
struct Segment_Tree
{
int A[N<<2],B[N<<2],sum[N<<2];
iv pp(int rt)
{
int u=min(A[lc],B[rc]);
sum[rt]=sum[lc]+sum[rc]+u;
A[rt]=A[lc]+A[rc]-u;
B[rt]=B[lc]+B[rc]-u;
}
iv insert(int rt,int l,int r,int p,int x,int y)
{
if(l==r)
{
A[rt]+=x;
B[rt]+=y;
return;
}
int mid=(l+r)>>1;
if(mid>=p)
insert(lc,l,mid,p,x,y);
else
insert(rc,mid+1,r,p,x,y);
pp(rt);
}
}T;
int main()
{
const int N=100000;
n=read();
for(re i=1;i<=n;i++)
{
a[i]=read();
T.insert(1,1,N,a[i],1,0);
}
for(re i=1;i<=n;i++)
{
b[i]=read();
T.insert(1,1,N,b[i],0,1);
sm.insert(b[i]);
}
int ans=T.sum[1];
for(re i=1;i<=n;i++)
{
int l=a[i]+1,r=*(--sm.end()),out=-1;
T.insert(1,1,N,a[i],-1,0);
while(l<=r)
{
int mid=(l+r)>>1;
T.insert(1,1,N,mid,0,-1);
if(T.sum[1]+1==ans)
{
l=mid+1;
out=mid;
}
else
r=mid-1;
T.insert(1,1,N,mid,0,1);
}
if(out!=-1)
{
--ans;
sm.erase(sm.find(out));
T.insert(1,1,N,out,0,-1);
printf("%d ",out);
}
else
{
l=1,r=a[i],out;
while(l<=r)
{
int mid=(l+r)>>1;
T.insert(1,1,N,mid,0,-1);
if(T.sum[1]==ans)
{
l=mid+1;
out=mid;
}
else
r=mid-1;
T.insert(1,1,N,mid,0,1);
}
printf("%d ",out);
sm.erase(sm.find(out));
T.insert(1,1,N,out,0,-1);
}
}
return 0;
}
T2 Time
这道题,或者说是这一种对数列进行排列的题,我是比较不自信的,因为自己不怎么有思路,导致我没有留出时间进行思考,这也是一个教训,逃避不是永久的办法,只有自己敢去想,去做,才能解决问题。
思路:这道题要求小的数字往两边靠,那很显然,如果要做到最小操作次数就要比较向左还是向右移动更优,可以利用线段树或者树状数组实现,算法的正确性在于,当我们将小数字向边缘移动的时候,我们都会将比他大的数字向中心移动,所以我们只需要利用贪心的思想只考虑使当前最优即可。
AC_code
#include<bits/stdc++.h>
#define re register int
#define ii inline int
#define iv inline void
#define lc (rt<<1)
#define rc (rt<<1|1)
#define mid ((l+r)>>1)
using namespace std;
const int N=1e5+10;
const int INF=1e9+10;
int n;
deque<int>q[N];
ii read()
{
int x=0;
bool f=1;
char ch=getchar();
while(ch<'0'||ch>'9')
{
if(ch=='-')
f=0;
ch=getchar();
}
while(ch>='0'&&ch<='9')
{
x=(x<<1)+(x<<3)+(ch^48);
ch=getchar();
}
return f?x:(-x);
}
struct Segment_Tree
{
int sum[N<<2];
iv pp(int rt)
{
sum[rt]=sum[lc]+sum[rc];
}
iv insert(int rt,int l,int r,int p,int z)
{
if(l==r)
{
sum[rt]+=z;
return;
}
if(mid>=p)
insert(lc,l,mid,p,z);
else
insert(rc,mid+1,r,p,z);
pp(rt);
}
ii query(int rt,int l,int r,int L,int R)
{
if(L>R)
return 0;
if(L<=l&&r<=R)
return sum[rt];
if(mid>=R)
return query(lc,l,mid,L,R);
if(mid<L)
return query(rc,mid+1,r,L,R);
return query(lc,l,mid,L,R)+query(rc,mid+1,r,L,R);
}
}T;
int main()
{
int ans=0;
n=read();
for(re i=1;i<=n;i++)
{
q[read()].push_back(i);
T.insert(1,1,n,i,1);
}
for(re i=1;i<=n;i++)
{
while(!q[i].empty())
{
int l=q[i].front(),r=q[i].back();
int sl=T.query(1,1,n,1,l-1),sr=T.query(1,1,n,1,n)-T.query(1,1,n,1,r);
if(sl<=sr)
{
ans+=sl;
T.insert(1,1,n,l,-1);
q[i].pop_front();
}
else
{
ans+=sr;
T.insert(1,1,n,r,-1);
q[i].pop_back();
}
}
}
printf("%d\n",ans);
return 0;
}
T3 Cover
好吧,这道题怪我没有认真听课,区间两两之间只有包含和不相交的关系
满足这个条件,他们的包含关系一定会构成一颗树,那么这道题显然就是一个树形DP(而不是我写的区间DP),记 \(f_{i,j}\)表示在 i 为根的子树中点被覆盖的最多次数为 j 的最优答案,转移
的时候首先将所有子树的答案直接合并,然后考虑点 i 的贡献 \(f_{i,j}=max(f_{i,j},f_{i,j-1}+val)\) , 这样朴素的DP方程显然时间复杂度会爆炸,考虑优化,我们考虑答案更新的过程,我们利用包含关系构建一颗树,那么我们将子树合并的时候,当前的节点会存储多个val
,包括自己本身的,还有自己的儿子合并得到的,这种答案属于同一层,我们需要不同层之间的转移。也就是到达跟节点的时候我们需要不断从左右区间中分别拿出最大值,这就是我们需要的答案,那么我们就可以利用一个 set ,通过一个合并的操作插入值并排序,具体实现见代码:
AC_code
#include<bits/stdc++.h>
#define re register int
#define ii inline int
#define iv inline void
#define lc (rt<<1)
#define rc (rt<<1|1)
#define mid ((l+r)>>1)
using namespace std;
const int N=1e5+10;
const int INF=1e9+10;
int n;
deque<int>q[N];
ii read()
{
int x=0;
bool f=1;
char ch=getchar();
while(ch<'0'||ch>'9')
{
if(ch=='-')
f=0;
ch=getchar();
}
while(ch>='0'&&ch<='9')
{
x=(x<<1)+(x<<3)+(ch^48);
ch=getchar();
}
return f?x:(-x);
}
struct Segment_Tree
{
int sum[N<<2];
iv pp(int rt)
{
sum[rt]=sum[lc]+sum[rc];
}
iv insert(int rt,int l,int r,int p,int z)
{
if(l==r)
{
sum[rt]+=z;
return;
}
if(mid>=p)
insert(lc,l,mid,p,z);
else
insert(rc,mid+1,r,p,z);
pp(rt);
}
ii query(int rt,int l,int r,int L,int R)
{
if(L>R)
return 0;
if(L<=l&&r<=R)
return sum[rt];
if(mid>=R)
return query(lc,l,mid,L,R);
if(mid<L)
return query(rc,mid+1,r,L,R);
return query(lc,l,mid,L,R)+query(rc,mid+1,r,L,R);
}
}T;
int main()
{
int ans=0;
n=read();
for(re i=1;i<=n;i++)
{
q[read()].push_back(i);
T.insert(1,1,n,i,1);
}
for(re i=1;i<=n;i++)
{
while(!q[i].empty())
{
int l=q[i].front(),r=q[i].back();
int sl=T.query(1,1,n,1,l-1),sr=T.query(1,1,n,1,n)-T.query(1,1,n,1,r);
if(sl<=sr)
{
ans+=sl;
T.insert(1,1,n,l,-1);
q[i].pop_front();
}
else
{
ans+=sr;
T.insert(1,1,n,r,-1);
q[i].pop_back();
}
}
}
printf("%d\n",ans);
return 0;
}
noip模拟测试31的更多相关文章
- [NOIP模拟测试31]题解
A.math 考场乱搞拿了95,2333. 考虑裴蜀定理:$ax+by=z$存在整数解,当且仅当$gcd(a,b)|z$. 那么如果某个数能够被拼出来,就必须满足所有$a_i$的$gcd$是它的因子. ...
- 「题解」NOIP模拟测试题解乱写II(36)
毕竟考得太频繁了于是不可能每次考试都写题解.(我解释个什么劲啊又没有人看) 甚至有的题目都没有改掉.跑过来写题解一方面是总结,另一方面也是放松了. NOIP模拟测试36 T1字符 这题我完全懵逼了.就 ...
- 2019.8.3 [HZOI]NOIP模拟测试12 C. 分组
2019.8.3 [HZOI]NOIP模拟测试12 C. 分组 全场比赛题解:https://pan.baidu.com/s/1eSAMuXk 刚看这题觉得很难,于是数据点分治 k只有1和2两种,分别 ...
- 2019.8.3 [HZOI]NOIP模拟测试12 B. 数颜色
2019.8.3 [HZOI]NOIP模拟测试12 B. 数颜色 全场比赛题解:https://pan.baidu.com/s/1eSAMuXk 数据结构学傻的做法: 对每种颜色开动态开点线段树直接维 ...
- 2019.8.3 [HZOI]NOIP模拟测试12 A. 斐波那契(fibonacci)
2019.8.3 [HZOI]NOIP模拟测试12 A. 斐波那契(fibonacci) 全场比赛题解:https://pan.baidu.com/s/1eSAMuXk 找规律 找两个节点的lca,需 ...
- NOIP模拟测试17&18
NOIP模拟测试17&18 17-T1 给定一个序列,选取其中一个闭区间,使得其中每个元素可以在重新排列后成为一个等比数列的子序列,问区间最长是? 特判比值为1的情况,预处理比值2~1000的 ...
- 「题解」NOIP模拟测试题解乱写I(29-31)
NOIP模拟29(B) T1爬山 简单题,赛时找到了$O(1)$查询的规律于是切了. 从倍增LCA那里借鉴了一点东西:先将a.b抬到同一高度,然后再一起往上爬.所用的步数$×2$就是了. 抬升到同一高 ...
- NOIP模拟测试1(2017081501)
好,今天是cgg第一次举行模拟测试,希望各位支持. 时间限制:2小时 题目链接: 题目一:水得都没名字了 题目二:车站 题目三:选数 不要觉得2小时太少,我的题目很良心,都很简单. 答案可以在模拟测试 ...
- 2019.8.14 NOIP模拟测试21 反思总结
模拟测试20的还没改完先咕着 各种细节问题=错失190pts T1大约三分钟搞出了式子,迅速码完,T2写了一半的时候怕最后被卡评测滚去交了,然后右端点没有初始化为n…但是这样还有80pts,而我后来还 ...
随机推荐
- 如何回答面试中问到的Hibernate和MyBatis的区别
这边主要是写给那些准备去面试的(没什么经验的)应聘者看的,为了在面试中更好的回答这个问题,我做一个简单的梳理和总结. 作为一名职场新人,经历过多次的面试,由于在简历中提及了Hibernate和MyBa ...
- 第15章: Prometheus监控Kubernetes资源与应用
Prometheus监控Kubernetes资源与应用 目录 1 监控方案 2 2 监控指标 4 3 实现思路 4 4 在K8S中部署Prometheus 4 5 在K8S中部署Grafana与可视化 ...
- MySQL数据库快速入门与应用实战(阶段一)
MySQL数据库快速入门与应用实战(阶段一) 作者 刘畅 时间 2020-09-02 实验环境说明: 系统:centos7.5 主机名 ip 配置 slavenode3 172.16.1.123 4核 ...
- 10 一键部署LNMP网站平台
#!/bin/bash export PATH=/usr/local/sbin:/usr/local/bin:/usr/sbin:/usr/bin:/root/bin NGINX_V=1.19.1 P ...
- 27、路由 route
优先于网卡中的配置,但是重启或重启网卡失效,最好加入到开机自启动服务中/etc/rc.local文件中: 27.1.添加网络路由: 目的网络.目的网络掩码,网关.网卡: 网络寻址以路由表中的路由优先: ...
- 14、iptables_nat源地址转换(内网共享上网)
14.1.环境说明: 1.架构图:
- 99、centos下安装teamviewer
99.1.teamviewer简介: TeamViewer是一个能在任何防火墙和NAT代理的后台用于远程控制的应用程序,桌面共享和文件传输的简单且快速的解决方案. 为了连接到另一台计算机,只需要在两台 ...
- 11、gitlab和Jenkins整合(2)
5.补充: (1)构建说明: 1)Jenkins会基于一些处理器任务后,构建发布一个稳健指数 (从0-100 ),这些任务一般以插件的方式实现. 2)它们可能包括单元测试(JUnit).覆盖率(Cob ...
- Flask(5)- 动态路由
前言 前面几篇文章讲的路由路径(rule)都是固定的,就是一个路径和一个视图函数绑定,当访问这条路径时会触发相应的处理函数 这样无法处理复杂的情况,比如常见的一个课程分类下有很多个课程,那么他们的 p ...
- WebSocket实现实时聊天系统
WebSocket实现实时聊天系统 等闲变却故人心,却道故人心易变. 简介:前几天看了WebSocket,今天体验下它的实时聊天. 一.项目介绍 WebSocket 实时聊天系统自己一个一码的搞出来还 ...