【XSY2732】Decalcomania 可持久化线段树 分治
题目描述
有一个陶瓷瓶周围有\(n\)个可以印花的位置。第\(i\)个与第\(i+1\)个位置之间的距离为\(d_i\),在第\(i\)个位置印图案要\(t_i\)秒。
机器刚开始在\(0\)号位置,可以以\(1\)单位长度每秒的速度移动。
一个位置只能印一个图案。
现在有\(m\)秒,问你最多能印几个图案。
保证时间不足以绕陶瓷瓶一圈。
\(n\leq 100000\)
题解
肯定是先往一边移动在移动到另外一边。
不妨设先往右边移动,那么右边的距离就要\(\times 2\)。
求出每边印\(i\)个印花最少要多少秒。
然后把两边合并即可。
考虑怎么求。
显然印\(i+1\)个印花的最优方案所需要移动的最右端点肯定在印\(i\)个印花的右边。
证明:考虑反证法。
设\(f(i,j)\)为印\(i\)个印花且最右端点为\(j\)的代价,\(a(i,j)\)为在前\(i\)个端点印印花所需要的第\(j\)短时间。
设印\(i\)个印花的最优方案所需要移动的最右端点为\(j\),\(i+1\)个的右端点是\(k(k<j)\)
那么有\(f(i,j)<f(i,k),a(j,i+1)<a(k,i+1),f(i+1,j)>f(i+1,k)\)
但是前两项加起来是第三项,所以不合法。
然后就可以分治了。
设当前要求印\(l\)个到\(r\)个的答案,答案区间为\([sl,sr]\)。
先求出印\(mid=\frac{l+r}{2}\)个的答案和右端点位置\(k\),可以通过枚举右端点得到。
\([l,mid-1]\)的右端点位置在\([sl,k]\),\([mid+1,r]\)的右端点位置在\([k,sr]\)
然后分治下去即可。
求前面\(i\)个位置最小的\(j\)个印印花的时间可以通过可持久化线段树得到。
时间复杂度:\(O(n\log^2n)\)
代码
#include<cstdio>
#include<cstring>
#include<algorithm>
#include<utility>
using namespace std;
typedef long long ll;
typedef pair<int,int> pii;
int q;
char cc[10000010];
int tt;
int h[100010];
ll rd()
{
ll s=0;
int c;
while((c=cc[tt++])<'0'||c>'9');
s=c-'0';
while((c=cc[tt++])>='0'&&c<='9')
s=s*10+c-'0';
return s;
}
namespace seg
{
int ls[4000010];
int rs[4000010];
ll s[4000010];
int sz[4000010];
int cnt;
int insert(int p1,int &x,int l,int r)
{
int p=++cnt;
ls[p]=ls[p1];
rs[p]=rs[p1];
s[p]=s[p1]+h[x];
sz[p]=sz[p1]+1;
if(l==r)
return p;
int mid=(l+r)>>1;
if(x<=mid)
ls[p]=insert(ls[p],x,l,mid);
else
rs[p]=insert(rs[p],x,mid+1,r);
return p;
}
ll query(int p,int x,int l,int r)
{
if(l==r)
return (ll)x*h[l];
int mid=(l+r)>>1;
int lsz=sz[ls[p]];
if(x<=lsz)
return query(ls[p],x,l,mid);
return s[ls[p]]+query(rs[p],x-lsz,mid+1,r);
}
}
int rt[100010];
int n;
ll m;
ll d[100010];
int t[100010];
ll a[100010];
ll f1[100010];
ll f2[100010];
ll g1[100010];
ll g2[100010];
int b[100010];
int c[100010];
int e[100010];
int f[100010];
ll &mm=m;
void gao(int l,int r,int sl,int sr,ll *s)
{
if(l>r)
return;
int mid=(l+r)>>1;
ll ans=0x3fffffffffffffffll;
int i;
int m=sr;
for(i=sl;i<=sr;i++)
if(i>=mid&&a[i-1]<=mm)
{
ll v=a[i-1]+seg::query(rt[i],mid,0,q);
if(v<ans)
{
ans=v;
m=i;
}
}
s[mid]=ans;
gao(l,mid-1,sl,m,s);
gao(mid+1,r,m,sr,s);
}
void gao(ll *s)
{
memset(rt,0,sizeof rt);
seg::cnt=0;
int i;
for(i=1;i<=n;i++)
rt[i]=seg::insert(rt[i-1],c[i],0,q);
gao(1,n,1,n,s);
}
int main()
{
#ifndef ONLINE_JUDGE
freopen("a.in","r",stdin);
freopen("a2.out","w",stdout);
#endif
fread(cc+1,10000000,1,stdin);
// scanf("%d%lld",&n,&m);
n=rd();
m=rd();
int i;
q=0;
for(i=1;i<=n;i++)
// scanf("%lld%lld",&d[i],&t[i]);
{
d[i]=rd();
t[i]=rd();
h[++q]=t[i];
}
sort(h+1,h+q+1);
q=unique(h+1,h+q+1)-h-1;
for(i=1;i<=n;i++)
t[i]=lower_bound(h+1,h+q+1,t[i])-h;
for(i=1;i<=n;i++)
{
a[i]=d[i];
c[i]=t[i];
}
for(i=1;i<=n;i++)
a[i]+=a[i-1];
gao(f1);
for(i=1;i<=n;i++)
a[i]*=2;
c[1]=0;
gao(g1);
reverse(t+2,t+n+1);
reverse(d+1,d+n+1);
for(i=1;i<=n;i++)
{
a[i]=d[i];
c[i]=t[i];
}
for(i=1;i<=n;i++)
a[i]+=a[i-1];
gao(f2);
for(i=1;i<=n;i++)
a[i]*=2;
c[1]=0;
gao(g2);
int ans=0;
for(i=1;i<=n;i++)
if(f1[i]<=m)
ans=max(ans,i);
for(i=1;i<=n;i++)
if(f2[i]<=m)
ans=max(ans,i);
int j;
j=n;
for(i=1;i<=n;i++)
{
while(j&&f1[i]+g2[j]>m)
j--;
if(!j)
break;
ans=max(ans,i+j-1);
}
j=n;
for(i=1;i<=n;i++)
{
while(j&&f2[i]+g1[j]>m)
j--;
if(!j)
break;
ans=max(ans,i+j-1);
}
printf("%d\n",ans);
return 0;
}
【XSY2732】Decalcomania 可持久化线段树 分治的更多相关文章
- BZOJ.4137.[FJOI2015]火星商店问题(线段树分治 可持久化Trie)
BZOJ 洛谷 一直觉得自己非常zz呢.现在看来是真的=-= 注意题意描述有点问题,可以看BZOJ/洛谷讨论. 每个询问有两个限制区间,一是时间限制\([t-d+1,t]\),二是物品限制\([L,R ...
- BZOJ4012[HNOI2015]开店——树链剖分+可持久化线段树/动态点分治+vector
题目描述 风见幽香有一个好朋友叫八云紫,她们经常一起看星星看月亮从诗词歌赋谈到 人生哲学.最近她们灵机一动,打算在幻想乡开一家小店来做生意赚点钱.这样的 想法当然非常好啦,但是她们也发现她们面临着一个 ...
- 2019.01.13 bzoj4137: [FJOI2015]火星商店问题(线段树分治+可持久化01trie)
传送门 题意:序列上有nnn个商店,有两种事件会发生: sss商店上进购标价为vvv的一个物品 求编号为[l,r][l,r][l,r]之间的位置买ddd天内新进购的所有物品与一个数xxx异或值的最大值 ...
- 【xsy2818】 最近点 动态树分治+可持久化线段树
题目大意:给你一颗n个节点的树,最初点集S为空. 有m次操作:往当前点集S中加入/删除一个点,询问点x至集合S中任意点的最小距离,回到第t次修改点集的操作后的状态. 数据范围:$n,m≤10^5$ 我 ...
- bzoj 4137 [FJOI2015]火星商店问题——线段树分治+可持久化01trie树
题目:https://www.lydsy.com/JudgeOnline/problem.php?id=4137 关于可持久化01trie树:https://www.cnblogs.com/LadyL ...
- 【BZOJ4137】火星商店问题(线段树分治,可持久化Trie)
[BZOJ4137]火星商店问题(线段树分治,可持久化Trie) 题面 洛谷 BZOJ权限题 题解 显然可以树套树,外层线段树,内层可持久化Trie来做. 所以我们需要更加优美的做法.--线段树分治. ...
- [FJOI2015]火星商店问题(线段树分治,可持久化,Trie树)
[FJOI2015]火星商店问题 前天考了到线段树分治模板题,全场都切了,就我不会QAQ 于是切题无数的Tyher巨巨就告诉我:"你可以去看看火星商店问题,看了你就会了." 第一道 ...
- [FJOI2015]火星商店问题(线段树分治+可持久化Trie)
重新写一年前抄题解的那题,当时我啥都不会只是Ctrl+C,Ctrl+V写过的题,今天重新写一遍. 题解: 不会线段树分治,还是学一下这东西吧,这是我的第一道线段树分治. 首先对于特殊商品,可以直接可持 ...
- 【洛谷P4585】 [FJOI2015]火星商店问题 线段树分治+可持久化trie
感觉这个线段树分治和整体二分几乎相同啊~ code: #include <bits/stdc++.h> #define MAX 100300 #define ll long long #d ...
随机推荐
- python-Selenium库的详解
一.什么是Selenium selenium 是一套完整的web应用程序测试系统,包含了测试的录制(selenium IDE),编写及运行(Selenium Remote Control)和测试的并行 ...
- Mergeable Stack(链表实现栈)
C - Mergeable Stack ZOJ - 4016 一开始用stl中内置的栈来写,其中第三个操作,我先复制到一个数组,再将其倒给另一个栈 这个方法有两个错误的地方: 1.栈在内存很大需要扩容 ...
- C++入门之初话多态与虚函数
多态性是面向对象程序设计的又一个重要思想,关于多态的详尽描述,请看本人的收藏https://www.cnblogs.com/hust-ghtao/p/3512461.html.这篇博文中,详尽的探讨了 ...
- c++入门之详细探讨类的一些行为
之前我们讨论过类成员的组成,尤其是成员函数,我们知道了定义一个类的时候,我们往往定义了:构造函数,析构函数,其他函数,以及友元函数(友元函数不是必须的). 同时,我们知道了这样一个事情:在定义一个对象 ...
- FPGA中边沿触发和电平触发
边沿触发和电平触发基本就是触发器和锁存器的区别: 触发器是边沿触发,只有当时钟上升(或下降)的一瞬间,触发器会读取并锁存输入信号.输出信号仅在时钟信号上升(或下降)的一瞬间会发生变化. 锁存器是电 ...
- 如果需要精确的答案,请避免使用float和double
Java中的简单浮点数类型float和double不能够进行运算.不光是Java,在其它很多编程语言中也有这样的问题.在大多数情况下,计算的结果是准确的,但是多试几次(可以做一个循环)就可以试出类似上 ...
- redis中的hash、列表、集合操作
一.hash操作 数据结构:key:{k1:v1, k2:v2, k3:v3} 类似Python中的字典 如:info : {name: lina, age: 22, sex: F} hset key ...
- tomcat8.0部署启动
http://tomcat.apache.org/download-80.cgi 打开命令行提示符窗口, 进入Tomcat安装目录, 进入bin目录下, 输入:service.bat install ...
- Freemarker 页面静态化技术使用入门案例
在访问 新闻.活动.商品 详情页面时, 路径可以是 xx[id].html, 服务器端根据请求 id, 动态生成 html 网页,下次访问数据时,无需再查询数据,直接将 html 静态页面返回.这样一 ...
- 在linux和本地系统之间进行数据传输的简单方法--lrzsz
lrzsz是一款在linux里可代替ftp上传和下载的程序. >>提君博客原创 http://www.cnblogs.com/tijun/ << 提君博客原创 安装和使用非 ...