【2020.11.19提高组模拟】二次剩余two 题解

题目描述

有\(n\)个二次函数,每个二次函数可以用两个值\(m,k\)描述:

\[f(x)=(x-m)^2+k
\]

现在有\(q\)次操作:

1.插入一个二次函数\((m,k)\)

2.\((x,t)\)删除所有\(f(x)\le t\)的二次函数。

输出每次操作后还剩下的二次函数个数。

对于所有的数据,保证\(n,q,m,k,x,t\in [1,1e5]\)

以下是数据范围表格节选。

\(n,q\le\) \(m\le\)
1~4 20000 100000
5~6 100000 1000
7~10 100000 100000

Solution

考场上我先想到了对\(n^2\)的暴力进行卡常优化——结果就过了qaq老师不要重测卡我啊!

虽然这不是正解,但是我还是记录一下吧。

用了两个栈,先把开始的二次函数存到栈1。

对于操作1,直接入栈1。

对于操作2,扫描栈1内所有的二次函数,若无需删去就加入栈2,否则不加;再将栈2复制到栈1。

可以利用类似滚动数组的方法在两个栈之间回滚,省去复制操作;不要暴力清空栈,只要把size赋值为0就好。这样每次进行操作2需要遍历的次数仅仅为当前二次函数的个数。

Code-\(O(n^2)\)

#include<iostream>
#include<algorithm>
#include<cstdio>
#include<cstring>
#include<cmath>
#include<map>
#include<set>
#include<queue>
#include<vector>
#define IL inline
#define re register
#define LL long long
#define ULL unsigned long long
#define debug printf("Now is %d\n",__LINE__);
using namespace std; template<class T>inline void read(T&x)
{
char ch=getchar();
int fu;
while(!isdigit(ch)&&ch!='-') ch=getchar();
if(ch=='-') fu=-1,ch=getchar();
x=ch-'0';ch=getchar();
while(isdigit(ch)){x=x*10+ch-'0';ch=getchar();}
x*=fu;
}
inline int read()
{
int x=0,fu=1;
char ch=getchar();
while(!isdigit(ch)&&ch!='-') ch=getchar();
if(ch=='-') fu=-1,ch=getchar();
x=ch-'0';ch=getchar();
while(isdigit(ch)){x=x*10+ch-'0';ch=getchar();}
return x*fu;
}
int G[55];
template<class T>inline void write(T x)
{
int g=0;
if(x<0) x=-x,putchar('-');
do{G[++g]=x%10;x/=10;}while(x);
for(re int i=g;i>=1;--i)putchar('0'+G[i]);putchar('\n');
}
int n,q;
struct two
{
LL m,k;
}sk[2][1000010];
int size[2];
int main()
{
freopen("two.in","r",stdin);
freopen("two.out","w",stdout);
n=read();
q=read();
for(re int i=1;i<=n;i++) sk[0][i].m=read(),sk[0][i].k=read();
size[0]=n;
int op,x,t;
bool flag=0;
while(q--)
{
op=read();
if(op==1)
{
sk[flag][++size[flag]].m=read();
sk[flag][size[flag]].k=read();
write(size[flag]);
}
else
{
x=read();
t=read();
for(re int i=1;i<=size[flag];i++)
{
if((x-sk[flag][i].m)*(x-sk[flag][i].m)+sk[flag][i].k>t) sk[!flag][++size[!flag]]=sk[flag][i];
}
size[flag]=0;
flag=!flag;
write(size[flag]);
}
}
return 0;
}

记得开\(\texttt{long long}\)。

Std

因为\(k\in \mathbb N^+\),所以先不考虑k。

\[f(x)=(x-m)^2\le t\\
\Rightarrow -\sqrt t \le x-m\le \sqrt t\\
\Rightarrow x-\sqrt t\le m\le x+\sqrt t
\]

所以,可能会被删去的二次函数的\(m\)只可能在\(x\pm\sqrt t\)中。

而对于相同的\(m\),k小的应该越会被删去。

所以我们对每种m都用一个小根堆,扫描时只要扫\(x-\sqrt t\le m\le x+\sqrt t\)区间中的堆,然后尽量取出合法的二次函数即可。

时间复杂度:\(O((n+q)\log n+q\sqrt t)\)。

Code

#include<iostream>
#include<algorithm>
#include<cstdio>
#include<cstring>
#include<cmath>
#include<map>
#include<set>
#include<queue>
#include<vector>
#define IL inline
#define re register
#define LL long long
#define ULL unsigned long long
#define debug printf("Now is %d\n",__LINE__);
using namespace std; template<class T>inline void read(T&x)
{
char ch=getchar();
int fu;
while(!isdigit(ch)&&ch!='-') ch=getchar();
if(ch=='-') fu=-1,ch=getchar();
x=ch-'0';ch=getchar();
while(isdigit(ch)){x=x*10+ch-'0';ch=getchar();}
x*=fu;
}
inline int read()
{
int x=0,fu=1;
char ch=getchar();
while(!isdigit(ch)&&ch!='-') ch=getchar();
if(ch=='-') fu=-1,ch=getchar();
x=ch-'0';ch=getchar();
while(isdigit(ch)){x=x*10+ch-'0';ch=getchar();}
return x*fu;
}
int G[55];
template<class T>inline void write(T x)
{
int g=0;
if(x<0) x=-x,putchar('-');
do{G[++g]=x%10;x/=10;}while(x);
for(int i=g;i>=1;--i)putchar('0'+G[i]);putchar('\n');
}
LL n,q,ans;
priority_queue<int>a[100010];
int main()
{
// freopen("two.in","r",stdin);
// freopen("two.out","w",stdout);
n=read();
q=read();
for(re int i=1,m,k;i<=n;i++) m=read(),k=read(),a[m].push(-k);
ans=n;
int op,x,y,l,r;
while(q--)
{
op=read();
if(op==1)
{
x=read();
y=read();
a[x].push(-y);
ans++;
}
else
{
x=read();
y=read();
l=x-sqrt(y);
r=x+sqrt(y);
for(re int i=max(l,1);i<=r&&i<=100000;i++)
{
while(!a[i].empty())
{
if((x-i)*(LL)(x-i)-a[i].top()>y) break;
a[i].pop();
ans--;
}
}
}
write(ans);
}
return 0;
}

Attention

不要抱有侥幸心理。这次能过,下次还是会\(40pts\)乃至\(20pts\)的。

【2020.11.19提高组模拟】二次剩余two 题解的更多相关文章

  1. 【2020.11.28提高组模拟】T1染色(color)

    [2020.11.28提高组模拟]T1染色(color) 题目 题目描述 给定 \(n\),你现在需要给整数 \(1\) 到 \(n\) 进行染色,使得对于所有的 \(1\leq i<j\leq ...

  2. 【2020.11.28提高组模拟】T2 序列(array)

    序列(array) 题目描述 ​给定一个长为 \(m\) 的序列 \(a\). 有一个长为 \(m\) 的序列 \(b\),需满足 \(0\leq b_i \leq n\),\(\sum_{i=1}^ ...

  3. 【2020.11.30提高组模拟】剪辣椒(chilli)

    剪辣椒(chilli) 题目描述 在花园里劳累了一上午之后,你决定用自己种的干辣椒奖励自己. 你有n个辣椒,这些辣椒用n-1条绳子连接在一起,任意两个辣椒通过用若干个绳子相连,即形成一棵树. 你决定分 ...

  4. 【2020.11.30提高组模拟】删边(delete)

    删边(delete) 题目 题目描述 给你一棵n个结点的树,每个结点有一个权值,删除一条边的费用为该边连接的两个子树中结点权值最大值之和.现要删除树中的所有边,删除边的顺序可以任意设定,请计算出所有方 ...

  5. JZOJ 【2020.11.30提高组模拟】剪辣椒(chilli)

    题目大意 给出一棵 \(n\) 个节点的树,删去其中两条边 使得分出的三个子树大小中最大与最小的差最小 分析 先一边 \(dfs\) 预处理出以 \(1\) 为根每个点的 \(size\) 然后按 \ ...

  6. JZOJ 6904. 【2020.11.28提高组模拟】T3 树上询问(query)

    题目 你有一棵 \(n\) 节点的树 ,回答 \(m\) 个询问,每次询问给你两个整数 \(l,r\) ,问存在多少个整数 \(k\) 使得从 \(l\) 沿着 \(l \to r\) 的简单路径走 ...

  7. 11.5NOIP2018提高组模拟题

    书信(letter) Description 有 n 个小朋友, 编号为 1 到 n, 他们每人写了一封信, 放到了一个信箱里, 接下来每个人从中抽取一封书信. 显然, 这样一共有 n!种拿到书信的情 ...

  8. 【2020.12.03提高组模拟】A组反思

    估计:40+10+0+0=50 实际:40+10+0+0=50 rank40 T1 赛时看到\(n,m\leq9\),我当机立断决定打表,暴力打了几个点之后发现在\(n\ne m\)且\(k\ne0\ ...

  9. 【2020.12.01提高组模拟】卡特兰数(catalan)

    题目 题目描述 今天,接触信息学不久的小\(A\)刚刚学习了卡特兰数. 卡特兰数的一个经典定义是,将\(n\)个数依次入栈,合法的出栈序列个数. 小\(A\)觉得这样的情况太平凡了.于是,他给出了\( ...

  10. 【2020.12.01提高组模拟】A组反思

    105,rk45 T1 赛时一开始先打了\(m=0\)的情况,也就是普通的卡特兰数,然后打了暴力,样例过了,把样例改改就不行了,原因没有保证是枚举的是合法的出栈序列 得分:\(WA\&TLE1 ...

随机推荐

  1. 自己修改的一款Typora学术主题Academic-zh-vq

    这款typora主题是在Academic-Zh主题的基础上修改而来的. 主题衍生路径: 官方Academic主题-->zh-academic主题-->Academic-Zh主题--> ...

  2. 编写你的第一个 Django 应用程序,第7部分

    本教程从教程 6 停止的地方开始.我们将继续使用网络投票应用程序,并将专注于自定义 Django 自动生成的管理站点,这是我们在教程 2 中首次探索的. 一.自定义管理表单 通过用 admin.sit ...

  3. MFC下北京时间与时间戳相互转换

    //时间戳转北京时间字串 CString getLocalDate(__int64 timestamp) { timestamp += 28800;//GTM偏移8个时区得到北京时间 tm p; gm ...

  4. Docker restart 重启容器

    就像很多常驻后台应用程序动不动可能就需要重启操作一样,有时候我们可能也需要重启容器 而重启容器使用的就是 docker restart 命令 docker restart <container_ ...

  5. Docker 容器跨主机多网段通信解决方案

    一.MacVlan实现Docker的跨主机网络通信的方案有很多,如之前博文中写到的通过部署 Consul服务实现Docker容器跨主机通信    Macvlan工作原理:    Macvlan是Lin ...

  6. [每日算法 - 华为机试] leetcode690. 员工的重要性

    入口 力扣(LeetCode)官网 - 全球极客挚爱的技术成长平台备战技术面试?力扣提供海量技术面试资源,帮助你高效提升编程技能,轻松拿下世界 IT 名企 Dream Offer.https://le ...

  7. 【Linux】5.3 Shell字符串

    Shell 字符串 字符串是shell编程中最常用最有用的数据类型(除了数字和字符串,也没啥其它类型好用了),字符串可以用单引号,也可以用双引号,也可以不用引号.单双引号的区别跟PHP类似. 1. 单 ...

  8. 【Linux】2.3 Linux目录结构

    基本介绍 linux 的文件系统是采用级层式的树状目录结构,在此结构中的最上层是根目录"/",然后在此目录下再创建其他的目录. 深刻理解 linux 树状文件目录是非常重要的,这里 ...

  9. AI数字人Heygem:口播与唇形同步的福音,无需docker,无需配置环境,一键整合包来了

    介绍 短短72小时,硅基智能在GitHub开源的数字人模型Heygem.ai便突破1,300颗Star,迅速成为全球技术社区瞩目的焦点.这一现象级的增长速度不仅彰显了Heygem.ai的技术领先性,更 ...

  10. Clion搭建C++开发环境

    1.下载和安装MinGW 1)下载链接:http://www.mingw.org/ 2)选择安装目录,目录尽可能简单(如:D:\MinGW)且不要包含中文和空格 3)添加相关的包 所需的包如下:min ...