题意:有一个长度为n的序列,让你把它分成k段,段内元素取or,段间取and,求能够得到的最大值。

这个算法是我和xz场上yy出来的,然而时间不够了没写出来,而且时间复杂度是$O(nlogn+nlogA)$的比官方题解都要低...(但是常数大了点)

设最大值为ans,我们假设S&ans=S,看看S能否用k条线段凑出来,则将原问题转化成了一个判定问题。从高到低一位一位地考虑,最多只需进行$O(logA)$次判定。

如何进行判定呢?

首先将原数组复制一倍接到后面,然后进行两次尺取。第一次求出每个左端点l所对应的能够覆盖S的最小的右端点r并把它作为一条线段放进数组里(能够覆盖S的意思是S的每一位上的1都可以在[l,r]区间里的某个元素中取到,可以用RMQ预处理区间or然后$O(1)$判断),第二次则对这些线段进行尺取,求出每条线段右边第一条和它不相交的线段,将每条线段与这样的线段连边,可以得到一棵树(或者森林,若为森林则将所有树和一个虚节点连边即可变成一棵树),只需要检查一下这棵树上是否有一个结点的l和与它距离为k的父亲结点的r的区间长度r-l+1是否小于n,从根节点dfs一遍即可。

代码:(我写了两份,第一份怕爆栈所以手写了数组模拟栈,第二份是普通的dfs)

 #include<bits/stdc++.h>
using namespace std;
typedef long long ll;
const int N=1e6+;
int n,k,a[N],ST[N][],Log[N],nl,hd[N],ne,q[N],tp;
void build() {
for(int i=; i<=*n; ++i)ST[i][]=a[i];
for(int k=; k<=Log[*n]; ++k)
for(int i=; i+(<<k)-<=*n; ++i)
ST[i][k]=ST[i][k-]|ST[i+(<<(k-))][k-];
}
int qry(int L,int R) {
int k=Log[R-L+];
return ST[L][k]|ST[R-(<<k)+][k];
}
struct D {int l,r;} line[N];
struct E {int v,nxt;} e[N];
struct ND {int u,dep;} sta[N];
void addedge(int u,int v) {e[ne]= {v,hd[u]},hd[u]=ne++;}
bool dfs() {
sta[tp=]= {nl,};
while(~tp) {
int u=sta[tp].u,dep=sta[tp--].dep;
q[dep]=u;
if(dep>=k&&line[q[dep-k+]].r-line[u].l+<=n)return ;
for(int i=hd[u]; ~i; i=e[i].nxt)sta[++tp]= {e[i].v,dep+};
}
return ;
}
bool ok(int S) {
nl=;
for(int i=,j=; i<=*n; ++i) {
if(j<i)j=i;
for(; j<=*n&&(qry(i,j)&S)!=S; ++j);
if(j<=*n)line[nl++]= {i,j};
}
for(int i=; i<=nl; ++i)hd[i]=-;
ne=;
for(int i=,j=; i<nl; ++i) {
for(; j<nl&&line[j].l<=line[i].r; ++j);
addedge(j,i);
}
return dfs();
}
int solve() {
int ret=;
for(int i=; i>=; --i)if(ok(ret|(<<i)))ret|=<<i;
return ret;
}
int main() {
Log[]=-;
for(int i=; i<N; ++i)Log[i]=Log[i>>]+;
scanf("%d%d",&n,&k);
for(int i=; i<=n; ++i)scanf("%d",&a[i]);
for(int i=; i<=n; ++i)a[i+n]=a[i];
build();
printf("%d\n",solve());
return ;
}
 #include<bits/stdc++.h>
using namespace std;
typedef long long ll;
const int N=1e6+;
int n,k,a[N],ST[N][],Log[N],nl,hd[N],ne,q[N];
void build() {
for(int i=; i<=*n; ++i)ST[i][]=a[i];
for(int k=; k<=Log[*n]; ++k)
for(int i=; i+(<<k)-<=*n; ++i)
ST[i][k]=ST[i][k-]|ST[i+(<<(k-))][k-];
}
int qry(int L,int R) {
int k=Log[R-L+];
return ST[L][k]|ST[R-(<<k)+][k];
}
struct D {int l,r;} line[N];
struct E {int v,nxt;} e[N];
void addedge(int u,int v) {e[ne]= {v,hd[u]},hd[u]=ne++;}
bool dfs(int u,int dep) {
q[dep]=u;
if(dep>=k&&line[q[dep-k+]].r-line[u].l+<=n)return ;
for(int i=hd[u]; ~i; i=e[i].nxt)if(dfs(e[i].v,dep+))return ;
return ;
}
bool ok(int S) {
nl=;
for(int i=,j=; i<=*n; ++i) {
if(j<i)j=i;
for(; j<=*n&&(qry(i,j)&S)!=S; ++j);
if(j<=*n)line[nl++]= {i,j};
}
for(int i=; i<=nl; ++i)hd[i]=-;
ne=;
for(int i=,j=; i<nl; ++i) {
for(; j<nl&&line[j].l<=line[i].r; ++j);
addedge(j,i);
}
return dfs(nl,);
}
int solve() {
int ret=;
for(int i=; i>=; --i)if(ok(ret|(<<i)))ret|=<<i;
return ret;
}
int main() {
Log[]=-;
for(int i=; i<N; ++i)Log[i]=Log[i>>]+;
scanf("%d%d",&n,&k);
for(int i=; i<=n; ++i)scanf("%d",&a[i]);
for(int i=; i<=n; ++i)a[i+n]=a[i];
build();
printf("%d\n",solve());
return ;
}

Kattis - bitwise Bitwise (RMQ+尺取+树上dfs)的更多相关文章

  1. hdu 4123 Bob’s Race 树的直径+rmq+尺取

    Bob’s Race Time Limit: 5000/2000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others) Probl ...

  2. HDU-4123-树形dp+rmq+尺取

    Bob’s Race Time Limit: 5000/2000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others)Total ...

  3. #333 Div2 Problem B Approximating a Constant Range (尺取 && RMQ || 尺取 && multiset)

    题目链接:http://codeforces.com/contest/602/problem/B 题意 :给出一个含有 n 个数的区间,要求找出一个最大的连续子区间使得这个子区间的最大值和最小值的差值 ...

  4. hdu4123-Bob’s Race(树形dp+rmq+尺取)

    题意:Bob想要开一个运动会,有n个房子和n-1条路(一棵树),Bob希望每个人都从不同的房子开始跑,要求跑的尽可能远,而且每条路只能走最多一次.Bob希望所有人跑的距离的极差不大于q,如果起点的编号 ...

  5. Codeforces 682C Alyona and the Tree (树上DFS+DP)

    题目链接:http://codeforces.com/problemset/problem/682/C 题目大意:取树上任意一个点v,若点v的子树中有一个点u使得dist(v,u)>a[u]那么 ...

  6. 玲珑OJ1088【蜜汁尺取】

    前言(膜法): 早上10点多开始膜的,然后到中午交了一发,感觉膜法不对啊!然后就兴起小窗了一发管理员,然后管理员给我发了in,out数据...可是太大并没有什么可取性... 还是自己试,然后发现自己搞 ...

  7. UVA - 11107 Life Forms (广义后缀自动机+后缀树/后缀数组+尺取)

    题意:给你n个字符串,求出在超过一半的字符串中出现的所有子串中最长的子串,按字典序输出. 这道题算是我的一个黑历史了吧,以前我的做法是对这n个字符串建广义后缀自动机,然后在自动机上dfs,交上去AC了 ...

  8. Codeforces - 6E - Exposition - 尺取

    https://codeforc.es/problemset/problem/6/E 既然可以多个log,那就直接map伺候.尺取之后要查询区间里面的最大值和最小值的差.众所周知尺取的时候要是不是有序 ...

  9. Gym 100703I---Endeavor for perfection(尺取)

    题目链接 http://codeforces.com/problemset/gymProblem/100703/I Description standard input/outputStatement ...

随机推荐

  1. 基于nodeJS的小说爬虫实战

    背景与需求分析 最近迷恋于王者荣耀.斗鱼直播与B站吃播视频,中毒太深,下班之后无心看书. 为了摆脱现状,能习惯看书,我开始看小说了,然而小说网站广告多而烦,屌丝心态不愿充钱,于是想到了爬虫. 功能分析 ...

  2. DataTable.Select筛选过滤数据返回DataRow[]转为DataTable添加到DataSet

    问题还原,如图所示,我们要筛选所有SHDP 为北京翠微KR的数据. 1. 筛选DataTable微软为我们提供了一个方法DataTable.Select(),其用法如下: 1)  Select()—— ...

  3. python基础知识(正则表达式)

    使用正则表示式分割字符串 split() re.split(pattern,string,[maxsplit],[flags]) re.split(指定一个模式字符串,要匹配的字符串,最大的拆分次数, ...

  4. Git本地初始化并推送到远程仓库

    git常用命令 1.全局配置git用户名邮箱 git config --global user.name '你的名字' git config --global user.email '你的邮箱地址' ...

  5. SQLServer2008R2复制发布订阅(含局域网跨网段)调试总结

    需要注意的问题点: 一.发布服务器A和订阅服务器B都在同一个网段的局域网内 1.两台服务器的SQLServer版本必须一致. 2.两服务器之间访问没难度,直接就可以搜索到. 3.注意请求订阅和推送订阅 ...

  6. Nginx网络负载均衡,负载均衡,网络负载,网络均衡

    本节就聊聊采用Nginx负载均衡之后碰到的问题: Session问题 文件上传下载 通常解决服务器负载问题,都会通过多服务器分载来解决.常见的解决方案有: 网站入口通过分站链接负载(天空软件站,华军软 ...

  7. Linux软链接硬链接的区别

    ln是linux中又一个非常重要命令,它的功能是为某一个文件在另外一个位置建立一个同步的链接.当我们需要在不同的目录,用到相同的文件时,我们不需要在每一个需要的目录下都放一个必须相同的文件,我们只要在 ...

  8. Python 入门 之 面向对象的三大特性(封装 / 继承 / 多态)

    Python 入门 之 面向对象的三大特性(封装 / 继承 / 多态) 1.面向对象的三大特性: (1)继承 ​ 继承是一种创建新类的方式,在Python中,新建的类可以继承一个或多个父类,父类又可以 ...

  9. 数据库oracle一些操作(MiTAC)

    oracle计算时间差函数: 两个Date类型字段:START_DATE,END_DATE,计算这两个日期的时间差(分别以天,小时,分钟,秒,毫秒): 天: ROUND(TO_NUMBER(END_D ...

  10. es分数_score衰减函数

    1.按日期衰变 GET news/doc/_search { "query" : { "function_score": { "query" ...