ZJOI讲课的题目,数据结构什么的还是很友好的说

首先我们发现题目中提到的距离\(\le L\)的东西显然可以用单调队列维护

但是暴力搞去不掉区间并的限制,那么我们考虑从区间并入手

对于这种问题的套路有一个就是线段树维护一个区间的最优解,然后计算完一个点的答案之后直接在线段树上更新即可

所以我们有了一个很naive的思路——线段树套单调队列,但随便一想时空复杂度都是\(O(n^2)\)的

让我们想一下复杂度变大的原因是什么,其实就是pushdown带来的大量空间浪费

我们再仔细观察依稀这个问题的性质,发现其可以标记永久化,那么就很舒服了,时空复杂度都达到了优秀的\(O(n\log\ n)\)

然后像我这样naive的人就写出了这样的巨慢CODE

#include<cstdio>
#include<cctype>
#include<deque>
#include<algorithm>
#define RI register int
#define CI const int&
#define Tp template <typename T>
using namespace std;
const int N=250005,INF=2e9;
int n,m,rst[N<<1],L[N],R[N],ans[N],dis[N],cnt,ret;
class FileInputOutput
{
private:
static const int S=1<<21;
#define tc() (A==B&&(B=(A=Fin)+fread(Fin,1,S,stdin),A==B)?EOF:*A++)
#define pc(ch) (Ftop<S?Fout[Ftop++]=ch:(fwrite(Fout,1,S,stdout),Fout[(Ftop=0)++]=ch))
char Fin[S],Fout[S],*A,*B; int Ftop,pt[15];
public:
Tp inline void read(T& x)
{
x=0; char ch; while (!isdigit(ch=tc()));
while (x=(x<<3)+(x<<1)+(ch&15),isdigit(ch=tc()));
}
Tp inline void write(T x)
{
if (!x) return (void)(pc('0'),pc('\n')); if (x<0) x=-x,pc('-'); RI ptop=0;
while (x) pt[++ptop]=x%10,x/=10; while (ptop) pc(pt[ptop--]+48); pc('\n');
}
inline void Fend(void)
{
fwrite(Fout,1,Ftop,stdout);
}
#undef tc
#undef pc
}F;
inline int find(CI x)
{
return lower_bound(rst+1,rst+cnt+1,x)-rst;
}
class Segment_Tree
{
private:
deque <int> dq[N<<3];
public:
#define TN CI now=1,CI l=1,CI r=cnt
#define O beg,end,pos
inline void build(TN)
{
dq[now].push_back(1); if (l==r) return; int mid=l+r>>1;
build(now<<1,l,mid); build(now<<1|1,mid+1,r);
}
inline void insert(CI beg,CI end,CI pos,TN)
{
while (!dq[now].empty()&&ans[pos]<ans[dq[now].back()]) dq[now].pop_back();
dq[now].push_back(pos); if (l==r) return; int mid=l+r>>1;
if (beg<=mid) insert(O,now<<1,l,mid); if (end>mid) insert(O,now<<1|1,mid+1,r);
}
inline void getpos(CI beg,CI end,CI pos,TN)
{
if (beg<=l&&r<=end)
{
while (!dq[now].empty()&&dis[pos]-dis[dq[now].front()]>m) dq[now].pop_front();
if (!dq[now].empty()&&(!~ret||ans[dq[now].front()]<ans[ret])) ret=dq[now].front(); return;
}
int mid=l+r>>1; if (beg<=mid) getpos(O,now<<1,l,mid); if (end>mid) getpos(O,now<<1|1,mid+1,r);
}
#undef TN
#undef O
}SEG;
int main()
{
//freopen("CODE.in","r",stdin); freopen("CODE.out","w",stdout);
RI i; for (F.read(n),F.read(m),i=2;i<=n;++i)
F.read(L[i]),F.read(R[i]),rst[++cnt]=L[i],rst[++cnt]=R[i],F.read(dis[i]);
sort(rst+1,rst+cnt+1); cnt=unique(rst+1,rst+cnt+1)-rst-1;
for (i=2;i<=n;++i) L[i]=find(L[i]),R[i]=find(R[i]);
for (SEG.build(),i=2;i<=n;++i)
{
ret=-1; SEG.getpos(L[i],R[i],i); if (!~ret) ans[i]=INF;
else ans[i]=ans[ret]+1; SEG.insert(L[i],R[i],i);
}
for (i=2;i<=n;++i) F.write(ans[i]!=INF?ans[i]:-1); return F.Fend(),0;
}

没办法,我们发现这个程序慢有两点:

  • deque巨慢无比,而且内存占用极大
  • 没有维护每个节点的答案,这样查询的时候复杂度极高

然后解决方案也很简单:

  • deque换成list(快如闪电)
  • 单独写删除操作,并且记下每个点的答案

然后就可以顺利地通过此题了QWQ

#include<cstdio>
#include<cctype>
#include<list>
#include<algorithm>
#define RI register int
#define CI const int&
#define Tp template <typename T>
using namespace std;
const int N=250005,INF=1e9;
int n,m,rst[N<<1],q[N],L[N],R[N],dis[N],ans[N],cnt,pos;
class FileInputOutput
{
private:
static const int S=1<<21;
#define tc() (A==B&&(B=(A=Fin)+fread(Fin,1,S,stdin),A==B)?EOF:*A++)
#define pc(ch) (Ftop<S?Fout[Ftop++]=ch:(fwrite(Fout,1,S,stdout),Fout[(Ftop=0)++]=ch))
char Fin[S],Fout[S],*A,*B; int Ftop,pt[15];
public:
Tp inline void read(T& x)
{
x=0; char ch; while (!isdigit(ch=tc()));
while (x=(x<<3)+(x<<1)+(ch&15),isdigit(ch=tc()));
}
Tp inline void write(T x)
{
if (!x) return (void)(pc('0'),pc('\n')); if (x<0) x=-x,pc('-'); RI ptop=0;
while (x) pt[++ptop]=x%10,x/=10; while (ptop) pc(pt[ptop--]+48); pc('\n');
}
inline void Fend(void)
{
fwrite(Fout,1,Ftop,stdout);
}
#undef tc
#undef pc
}F;
inline int find(CI x)
{
return lower_bound(rst+1,rst+cnt+1,x)-rst;
}
class Segment_Tree
{
private:
list <int> dq[N<<3]; int val[N<<3];
inline void miner(int &x,CI y)
{
if (y<x) x=y;
}
inline int get(CI now)
{
if (dq[now].empty()) return INF; return ans[dq[now].front()];
}
inline void pushup(CI now,const bool& op)
{
val[now]=get(now); if (op) miner(val[now],val[now<<1]),miner(val[now],val[now<<1|1]);
}
public:
#define TN CI now=1,CI l=1,CI r=cnt
#define O beg,end,pos
inline void build(TN)
{
val[now]=INF; if (l==r) return; int mid=l+r>>1; build(now<<1,l,mid); build(now<<1|1,mid+1,r);
}
inline void insert(CI beg,CI end,CI pos,TN)
{
if (beg<=l&&r<=end)
{
while (!dq[now].empty()&&ans[pos]<=ans[dq[now].back()])
dq[now].pop_back(); dq[now].push_back(pos); return pushup(now,l!=r);
}
int mid=l+r>>1; if (beg<=mid) insert(O,now<<1,l,mid);
if (end>mid) insert(O,now<<1|1,mid+1,r); pushup(now,l!=r);
}
inline void remove(CI beg,CI end,CI pos,TN)
{
if (beg<=l&&r<=end)
{
while (!dq[now].empty()&&dq[now].front()<=pos)
dq[now].pop_front(); return pushup(now,l!=r);
}
int mid=l+r>>1; if (beg<=mid) remove(O,now<<1,l,mid);
if (end>mid) remove(O,now<<1|1,mid+1,r); pushup(now,l!=r);
}
inline int query(CI beg,CI end,TN)
{
if (beg<=l&&r<=end) return val[now]; int mid=l+r>>1,ret=get(now);
if (beg<=mid) miner(ret,query(beg,end,now<<1,l,mid));
if (end>mid) miner(ret,query(beg,end,now<<1|1,mid+1,r)); return ret;
}
#undef TN
#undef O
}SEG;
int main()
{
//freopen("CODE.in","r",stdin); freopen("CODE.out","w",stdout);
RI i,H=1,T=1; for (F.read(n),F.read(m),i=2;i<=n;++i)
F.read(L[i]),F.read(R[i]),rst[++cnt]=L[i],rst[++cnt]=R[i],F.read(dis[i]);
sort(rst+1,rst+cnt+1); cnt=unique(rst+1,rst+cnt+1)-rst-1;
for (i=2;i<=n;++i) L[i]=find(L[i]),R[i]=find(R[i]);
for (SEG.build(),SEG.insert(L[1]=q[1]=1,R[1]=cnt,1),i=2;i<=n;++i)
{
while (H<=T&&dis[i]-dis[q[H]]>m) pos=q[H++],SEG.remove(L[pos],R[pos],pos);
ans[i]=SEG.query(L[i],R[i]); if (ans[i]!=INF)
F.write(++ans[i]),SEG.insert(L[i],R[i],i),q[++T]=i; else F.write(-1);
}
return F.Fend(),0;
}

BZOJ 1171: 大sz的游戏的更多相关文章

  1. bzoj 1171 大sz的游戏& 2892 强袭作战 (线段树+单调队列+永久性flag)

    大sz的游戏 Time Limit: 50 Sec  Memory Limit: 357 MBSubmit: 536  Solved: 143[Submit][Status][Discuss] Des ...

  2. 【BZOJ-2892&1171】强袭作战&大sz的游戏 权值线段树+单调队列+标记永久化+DP

    2892: 强袭作战 Time Limit: 50 Sec  Memory Limit: 512 MBSubmit: 45  Solved: 30[Submit][Status][Discuss] D ...

  3. [BZOJ1171][BZOJ2892]大sz的游戏

    [BZOJ1171][BZOJ2892]大sz的游戏 试题描述 大sz最近在玩一个由星球大战改编的游戏.话说绝地武士当前共控制了N个星球.但是,西斯正在暗处悄悄地准备他们的复仇计划.绝地评议会也感觉到 ...

  4. BZOJ1171: 大sz的游戏&BZOJ2892: 强袭作战

    Description 大sz最近在玩一个由星球大战改编的游戏.话说绝地武士当前共控制了N个星球.但是,西斯正在暗处悄悄地准备他们的复仇计划.绝地评议会也感觉到了这件事.于是,准备加派绝地武士到各星球 ...

  5. 【BZOJ1171】大sz的游戏(线段树+单调队列)

    点此看题面 大致题意: 有\(n\)个点,两点间最大通讯距离为\(L\).已知除\(1\)号点外第\(i\)个点能够发出和接收的信号区间\([l_i,r_i]\)以及到\(1\)号点的距离\(dis_ ...

  6. BZOJ1171 : 大sz的游戏

    f[i]=min(f[j])+1,线段j与线段i有交,且l[i]-l[j]<=L. 线段j与线段i有交等价于y[j]>=x[i],x[j]<=y[i]. 因为l[i]递增,所以可以维 ...

  7. BZOJ 3684 大朋友和多叉树

    BZOJ 3684 大朋友和多叉树 Description 我们的大朋友很喜欢计算机科学,而且尤其喜欢多叉树.对于一棵带有正整数点权的有根多叉树,如果它满足这样的性质,我们的大朋友就会将其称作神犇的: ...

  8. BZOJ 1444:[JSOI2009]有趣的游戏

    BZOJ 1444:[JSOI2009]有趣的游戏 题目链接 首先我们建出Trie图,然后高斯消元. 我们设\(f_i\)表示经过第\(i\)个点的期望次数: \[ f_x=\sum i\cdot p ...

  9. [BZOJ 3652]大新闻

    [BZOJ 3652] 大新闻 题意 随机从 \([0,n)\) 中选取一个整数 \(x\), 并从 \([0,n)\) 中再选取一个整数 \(y\). 有 \(p\) 的概率选取一个能令 \(x\o ...

随机推荐

  1. 统一修改表单参数(表单提交的空字符串统一转null)

    统一修改表单参数(表单提交的空字符串统一转null) 1.介绍: 我们业务中有时会遇到提交的表单中某个参数为空字符串,导致后台接受的为空字符串("")而不是我们理想中的null,会 ...

  2. python-memcached学习笔记

    介绍: memcached是免费.开源.高性能.分布式内存对象的缓存系统(键/值字典),旨在通过减轻数据库负载加快动态web应用程序的使用. 数据类型:只用一种字符串类型 1:安装 sudo apt- ...

  3. Hadoop系列009-NameNode工作机制

    本人微信公众号,欢迎扫码关注! NameNode工作机制 1 NameNode & SecondaryNameNode工作机制 1.1 第一阶段:namenode启动 1)第一次启动namen ...

  4. 使用 Moq 测试.NET Core 应用 -- 其它

    第一篇文章, 关于Mock的概念介绍: https://www.cnblogs.com/cgzl/p/9294431.html 第二篇文章, 关于方法Mock的介绍: https://www.cnbl ...

  5. .NET Core TDD 前传: 编写易于测试的代码 -- 依赖项

    第1篇: 讲述了如何创造"缝".  "缝"(seam)是需要知道的概念. 第2篇, 避免在构建对象时写出不易测试的代码. 本文是第3篇, 讲述依赖项和迪米特法则 ...

  6. 网络协议 17 - HTTPDNS:私人定制的 DNS 服务

    [前五篇]系列文章传送门: 网络协议 12 - HTTP 协议:常用而不简单 网络协议 13 - HTTPS 协议:加密路上无尽头 网络协议 14 - 流媒体协议:要说爱你不容易 网络协议 15 - ...

  7. 【反编译系列】三、反编译神器(jadx)

    版权声明:本文为HaiyuKing原创文章,转载请注明出处! 概述 今天在看玩Android网站,搜索反编译的时候,才发现有个更好用的反编译工具.特此记录下. 下载 http://www.wanand ...

  8. KnockoutJS-绑定元素

    工作变更,又走回了WPF,一个来月没有接触web开发了,之前的KnockoutJS却不想放弃,继续进行知识的巩固,下个月开始重新走回web开发之路,还是得用回一些习惯了的工具.本次开始接触各绑定元素功 ...

  9. 【大数据安全】基于Kerberos的大数据安全验证方案

    1.背景 互联网从来就不是一个安全的地方.很多时候我们过分依赖防火墙来解决安全的问题,不幸的是,防火墙是假设"坏人"是来自外部的,而真正具有破坏性的攻击事件都是往往都是来自于内部的 ...

  10. [SpringBoot guides系列翻译]调用RESTfulWebService

    原文 参考链接 CommandLineRunner Bean 翻译如何调用RESTful WebService 这节将演示如何在SpringBoot里面调用RESTful的WebService. 构建 ...