Portal --> agc012

Description

  有一排点,两点间有一定距离,初始的时候有一个行走值\(v\),如果说两点间距离不超过\(v\),那么可以在这两点间自由行走,如果当前\(v>0\)那么可以选择突然出现在任意一点,但是这样做之后\(v\)会减半(下取整),问每个位置出发能否到达所有的位置至少一次

  

Solution

  额其实感觉关键还是模型的转化

​  其实我们可以比较形象地将每个\(v\)(就是不停除以\(2\)直到\(0\),中途得到的那堆\(v\)),看成“一层”,具体举个例子的话就是:当\(v=4\)的时候,第一层为\(v=4\),第二层为\(v=2\),第三层为\(v=1\),第四层为\(v=0\),这样

​  然后每次突然出现其实就相当于往上走了一层,而每层中,有一些区间是可以在区间内随便走的,我们将这些区间看成若干条线段,那么现在的问题就变成了,每层选一条线段,问是否能够覆盖整个区间

​  想不到吧然后因为层数上限是\(19\)所以。。这是一道状压dp的题目==

​  

​  所以首先,我们先将每一层中的线段处理出来,将第\(i\)层的线段存在\(seg[i]\)数组里面,只用存右端点即可,左端点可以由前一条线段的右端点\(+1\)得到

​  考虑一下大概要用什么样的方式统计答案,我们可以考虑枚举第一层的每条线段\((l_i,r_i)\),如果说存在一种方案使得从\(1\)开始覆盖到一个\(>=l_i-1\)的位置,从\(n\)开始覆盖到一个\(<=r_i+1\)的位置,那么这段线段中的每一个点都是\(Possible\)的,否则就是\(Impossible\)

​  那么所以,我们考虑维护两个数组\(tol\)和\(tor\),其中\(tor[i]\)表示线段选择状态(如果这层选了线段那么对应的二进制位为\(1\)否则为\(0\))为\(i\)的情况下,从\(1\)开始往右最远能覆盖到的位置,\(tol[i]\)表示从\(n\)开始往左最远能覆盖到的位置,注意,因为最后统计答案的时候,第一层的线段是要枚举的,所以我们在计算\(tor\)和\(tol\)的枚举状态的时候,不能包含第一层

​  接下来定义两个过程:\(expandReft(which,x)\)表示在第\(which\)层的线段中,严格大于\(x\)的第一个右端点,\(expandLight(which,x)\)表示在第\(which\)层的线段中,严格小于\(x\)的最靠近\(n\)的右端点

​  那么我们可以得到转移式子:

\[\begin{aligned}
tor[st|St(i)]&=max(tor[st|St(i)],expandRight(i,tor[st]))\\
tol[st|St(i)]&=min(tol[st|St(i)],expandLeft(i,tol[st]-1))\\
\end{aligned}
\]

  至于为什么要用\(tol[st]-1\)的话。。是因为如果直接找\(tol[st]\)的话,我们可能找到\(tol[st]\),这就说明有一段以\(tol[st]-1\)为右端点的区间以及一段以\(tol[st]\)为左端点的区间,然而这个时候我们应该找前者而不是后者会出些小问题==(不过也可能只是实现方式的问题。。具体都是看二分查找要怎么写吧都是一些实现上的细节问题qwq)

  那么最后一个小问题就是,如果说第一层有很多条线段的话那就很凉凉,然而实际上,我们发现如果说第一层的线段条数\(>logv+1\)的话。。就根本不可能覆盖完了(层数不够),所以我们可以在计算之前先判一下这种全部都是\(Impossible\)的情况

  那所以我们的总复杂度就是\(O(nlogv+vlognlogv)\)了

  

  代码大概长这个样子

#include<iostream>
#include<cstdio>
#include<cstring>
#include<cmath>
using namespace std;
const int N=2*(1e5)+10,ST=1<<20,TOP=20,inf=2e9;
int seg[TOP+1][N],d[N],tor[ST],tol[ST],a[N];
int n,m,v,lg,all;
bool in(int st,int x){return st>>x-1&1;}
int St(int x){return 1<<x-1;}
void get_seg(){
for (int i=1;i<=lg;++i){
seg[i][0]=0;
for (int j=1;j<=n;++j){
if (j==1||d[j-1]>v>>i-1) ++seg[i][0];
seg[i][seg[i][0]]=j;
}
}
}
bool firstcheck(){
if (seg[1][0]<=lg) return true;
for (int i=1;i<=n;++i) printf("Impossible\n");
return false;
}
int expand_right(int which,int x){
int l=1,r=seg[which][0],mid,ret=r;
while (l<=r){
mid=l+r>>1;
if (seg[which][mid]>x) ret=mid,r=mid-1;
else l=mid+1;
}
return seg[which][ret];
}
int expand_left(int which,int x){
int l=1,r=seg[which][0],mid,ret=l;
while (l<=r){
mid=l+r>>1;
if (seg[which][mid]<x) ret=mid,l=mid+1;
else r=mid-1;
}
return seg[which][ret]+1;
}
void dp(){
all=1<<lg;
for (int st=0;st<all;++st) tor[st]=0,tol[st]=n+1;
for (int st=0;st<all;st+=2){
for (int i=2;i<=lg;++i){
if (in(st,i)) continue;
tor[st|St(i)]=max(tor[st|St(i)],expand_right(i,tor[st]));
tol[st|St(i)]=min(tol[st|St(i)],expand_left(i,tol[st]-1));
}
}
}
void get_ans(){
int L,R;
bool ok;
for (int i=1;i<=seg[1][0];++i){
L=i==1?1:seg[1][i-1]+1; R=seg[1][i];
ok=false;
for (int st=0;st<all&&!ok;st+=2)
if (L-1<=tor[st]&&tol[(all-1)-st-1]<=R+1) ok=true;
for (int j=L;j<=R;++j) printf(ok?"Possible\n":"Impossible\n");
}
} int main(){
#ifndef ONLINE_JUDGE
freopen("a.in","r",stdin);
#endif
scanf("%d%d",&n,&v);
for (int i=1;i<=n;++i) scanf("%d",a+i);
d[n]=inf;
for (int i=1;i<n;++i) d[i]=a[i+1]-a[i];
lg=0;
while ((v>>lg)>0) ++lg;
++lg;
get_seg();
if (!firstcheck()) return 0;
dp();
get_ans();
}

【agc012E】Camel and Oases的更多相关文章

  1. 【AGC012E】 Camel and Oases ST表+状压dp

    题目大意:一排点,两点间有距离. 初始你有一个行走值$v$,如果相邻两点距离不超过$v$你可以自由在这两点行走. 当$v$大于$0$时,你可以选择某一时刻突然飞到任意点,这样做后$v$会减半(下取整) ...

  2. 【AtCoder】【模拟】【模型转化】Camel and Oases(AGC012)

    题意: 有一个骆驼,n个绿洲遍布在数轴上,第i个绿洲的坐标为x[i],保证x[i]单增.骆驼的驼峰有体积初始值V.当驼峰的体积变为v的时候,驼峰中至多只能够存储v L的水.骆驼希望走完所有的绿洲,并且 ...

  3. 【Mybatis】MyBatis配置文件的使用(二)

    本例在[Mybatis]MyBatis快速入门(一)基础上继续学习XML映射配置文件 MyBatis 的配置文件包含了会深深影响 MyBatis 行为的设置(settings)和属性(properti ...

  4. Python高手之路【六】python基础之字符串格式化

    Python的字符串格式化有两种方式: 百分号方式.format方式 百分号的方式相对来说比较老,而format方式则是比较先进的方式,企图替换古老的方式,目前两者并存.[PEP-3101] This ...

  5. 【原】谈谈对Objective-C中代理模式的误解

    [原]谈谈对Objective-C中代理模式的误解 本文转载请注明出处 —— polobymulberry-博客园 1. 前言 这篇文章主要是对代理模式和委托模式进行了对比,个人认为Objective ...

  6. 【原】FMDB源码阅读(三)

    [原]FMDB源码阅读(三) 本文转载请注明出处 —— polobymulberry-博客园 1. 前言 FMDB比较优秀的地方就在于对多线程的处理.所以这一篇主要是研究FMDB的多线程处理的实现.而 ...

  7. 【原】Android热更新开源项目Tinker源码解析系列之一:Dex热更新

    [原]Android热更新开源项目Tinker源码解析系列之一:Dex热更新 Tinker是微信的第一个开源项目,主要用于安卓应用bug的热修复和功能的迭代. Tinker github地址:http ...

  8. 【调侃】IOC前世今生

    前些天,参与了公司内部小组的一次技术交流,主要是针对<IOC与AOP>,本着学而时习之的态度及积极分享的精神,我就结合一个小故事来初浅地剖析一下我眼中的“IOC前世今生”,以方便初学者能更 ...

  9. Python高手之路【三】python基础之函数

    基本数据类型补充: set 是一个无序且不重复的元素集合 class set(object): """ set() -> new empty set object ...

随机推荐

  1. html表单总结

    总结了下html表单: <!DOCTYPE html> <html lang="en"> <head> <meta charset=&qu ...

  2. 第三篇 Python关于mysql的API--pymysql模块, mysql事务

    python关于mysql的API--pymysql模块 pymysql是Python中操作MySQL的模块,其使用方法和py2的MySQLdb几乎相同. 模块安装 pip install pymys ...

  3. throttle(节流)和debounce(防抖)

    防抖和节流都是用来控制频繁调用的问题,但是这两种的应用场景是有区别的. throttle(节流) 有一个调用周期,在一个很长的时间里分为多段,每一段执行一次.例如onscroll,resize,500 ...

  4. SQL Server变量杂谈

    学习SQL的过程有进步的话还是一件很美妙的事情的 在第一家公司虽然只呆了两年,但是感觉是我进步最快的两年.那时候工作和生活都挺充实的,每天都有一点点的收获和付出,其中最大的收获莫过于掌握一些核心技能. ...

  5. eclipse 简单操作

    1.设置字体: windows --> preferences --> general --> Colors and Fonts --> Basic --> 双击 Tex ...

  6. 用 Python 构建一个极小的区块链

    虽然有些人认为区块链是一个早晚会出现问题的解决方案,但是毫无疑问,这个创新技术是一个计算机技术上的奇迹.那么,究竟什么是区块链呢? 区块链 以比特币(Bitcoin)或其它加密货币按时间顺序公开地记录 ...

  7. Scrum立会报告+燃尽图(十一月十八日总第二十六次):功能开发与讨论贡献分配规则

    此作业要求参见:https://edu.cnblogs.com/campus/nenu/2018fall/homework/2284 项目地址:https://git.coding.net/zhang ...

  8. 04慕课网《进击Node.js基础(一)》HTTP讲解

    HTTP:通信协议 流程概述: http客户端发起请求,创建端口默认8080 http服务器在端口监听客户端请求 http服务器向客户端返回状态和内容 稍微详细解析: 1.域名解析:浏览器搜素自身的D ...

  9. lintcode-205-区间最小数

    205-区间最小数 给定一个整数数组(下标由 0 到 n-1,其中 n 表示数组的规模),以及一个查询列表.每一个查询列表有两个整数 [start, end]. 对于每个查询,计算出数组中从下标 st ...

  10. IDEA中Git的更新/提交/还原方法

    记录一下在IDEA上怎样将写的代码提交到GitHub远程库: 下面这个图是基本的提交代码的顺序: 1. 将代码Add到stage暂存区本地修改了代码后,需先将代码add到暂存区,最后才能真正提价到gi ...