CodeForces - 1100F:Ivan and Burgers (线性基&贪心)(离线 在线)

时间:2021-02-13 14:27:05

题意:给定N个数,Q次询问,求区间最大异或和。

思路:一开始想的线性基+线段树。单次线性基合并的复杂度为20*20,结合线段树,复杂度为O(NlogN*20*20);显然,超时。

超时代码:

#include<bits/stdc++.h>
#define pb push_back
#define rep(i,a,b) for(int i=a;i<=b;i++)
#define rep2(i,a,b) for(int i=a;i>=b;i--)
using namespace std;
const int maxn=;
int a[maxn]; vector<int>G[maxn];
void read(int &x){
x=; char c=getchar();
while(c>''||c<'') c=getchar();
while(c>=''&&c<='') x=x*+c-'',c=getchar();
}
void add(vector<int>&Now,vector<int>&p)
{
rep(i,,){
int x=p[i]; if(!x) continue;
rep2(j,,){
if(x&(<<j)){
if(Now[j]) x^=Now[j];
else { Now[j]=x;break;}
}
}
}
}
void build(int Now,int L,int R)
{
rep(i,,) G[Now].pb();
if(L==R){
int x=a[L]; if(!x) return;
rep2(j,,){
if(x&(<<j)){
if(G[Now][j]) x^=G[Now][j];
else { G[Now][j]=x;break;}
}
}
return ;
}
int Mid=(L+R)>>;
build(Now<<,L,Mid); build(Now<<|,Mid+,R);
G[Now]=G[Now<<]; add(G[Now],G[Now<<|]);
}
void query(int Now,int L,int R,int l,int r,vector<int>& res)
{
if(l<=L&&r>=R) { res=G[Now]; return ;}
int Mid=(L+R)>>;
rep(i,,) res.pb();
if(l<=Mid){
vector<int>t;
query(Now<<,L,Mid,l,r,t);
res=t;
}
if(r>Mid) {
vector<int>t;
query(Now<<|,Mid+,R,l,r,t);
add(res,t);
}
}
int main()
{
int N,M,L,R; scanf("%d",&N);
rep(i,,N) read(a[i]);
build(,,N);
scanf("%d",&M);
while(M--){
read(L); read(R);
vector<int>t;
query(,,N,L,R,t);
int res=; rep2(i,,) if((res^t[i])>res) res^=t[i];
printf("%d\n",res);
}
return ;
}

我们考虑离线,把所有询问按右端点排序,然后从左到有处理询问,对于当前询问[L,R];我们把[1,R]所有的数加入线性基,关键是对于每一位,我们保留其为位置,这里肯定是贪心地保留越后面的位置越优。 那么查询的时候,如果一个线性基里的数位置>=L,则可以考虑更新答案。

#include<bits/stdc++.h>
#define rep(i,a,b) for(int i=a;i<=b;i++)
#define rep2(i,a,b) for(int i=a;i>=b;i--)
using namespace std;
const int maxn=;
struct in{
int l,r,id;
friend bool operator< (in w,in v){ return w.r<v.r;}
}s[maxn];
int N,Q,ans[maxn],a[maxn],p[],pos[];
void add(int x,int id)
{
rep2(i,,)
if(x&(<<i)){
if(!p[i]){
p[i]=x; pos[i]=id;
return ;
}
if(pos[i]<id) swap(p[i],x),swap(pos[i],id);
x^=p[i];
}
}
int query(int id)
{
int res=;
rep2(i,,) if(pos[i]>=id&&(res^p[i])>res) res^=p[i];
return res;
}
int main()
{
scanf("%d",&N);
rep(i,,N) scanf("%d",&a[i]);
scanf("%d",&Q);
rep(i,,Q) scanf("%d%d",&s[i].l,&s[i].r),s[i].id=i;
sort(s+,s+Q+); int L=;
rep(i,,Q){
while(L<=s[i].r&&L<=N) add(a[L],L),++L;
ans[s[i].id]=query(s[i].l);
}
rep(i,,Q) printf("%d\n",ans[i]);
return ;
}

那么同理,不难想出在线的做法,我们纪录一个前缀和 线性基,任然保留最大的位置。

#include<bits/stdc++.h>
#define rep(i,a,b) for(int i=a;i<=b;i++)
#define rep2(i,a,b) for(int i=a;i>=b;i--)
using namespace std;
const int maxn=;
int p[maxn][],pos[maxn][];
int main()
{
int N,Q,L,R,x;
scanf("%d",&N);
rep(i,,N) {
rep(j,,) p[i][j]=p[i-][j],pos[i][j]=pos[i-][j];
scanf("%d",&x); int ti=i;
rep2(j,,){
if(x&(<<j)){
if(!p[i][j]) { p[i][j]=x; pos[i][j]=ti; break; }
if(pos[i][j]<ti) swap(p[i][j],x),swap(pos[i][j],ti);
x^=p[i][j];
}
}
}
scanf("%d",&Q);
rep(i,,Q) {
scanf("%d%d",&L,&R);
int res=;
rep2(j,,) if(pos[R][j]>=L&&(res^p[R][j])>res) res^=p[R][j];
printf("%d\n",res);
}
return ;
}

CodeForces - 1100F:Ivan and Burgers (线性基&贪心)(离线 在线)的更多相关文章

  1. codeforces 1100F Ivan and Burgers 线性基 离线

    题目传送门 题意: 给出 n 个数,q次区间查询,每次查询,让你选择任意个下标为 [ l , r ] 区间内的任意数,使这些数异或起来最大,输出最大值. 思路:离线加线性基. 线性基学习博客1 线性基 ...

  2. CodeForces 1100F Ivan and Burgers

    CodeForces题面 Time limit 3000 ms Memory limit 262144 kB Source Codeforces Round #532 (Div. 2) Tags da ...

  3. BZOJ 2460 &amp&semi; 洛谷 P4570 &lbrack;BJWC2011&rsqb;元素 &lpar;线性基 贪心&rpar;

    题目链接: 洛谷 BZOJ 题意 给定 \(n\) 个矿石,每个矿石有编号和魔力值两种属性,选择一些矿石,使得魔力值最大且编号的异或和不为 0. 思路 线性基 贪心 根据矿石的魔力值从大到小排序. 线 ...

  4. Codeforces 1100F&lpar;线性基&plus;贪心&rpar;

    题目链接 题意 给定序列,$q(1\leq q \leq 100000) $次询问,每次查询给定区间内的最大异或子集. 思路 涉及到最大异或子集肯定从线性基角度入手.将询问按右端点排序后离线处理询问, ...

  5. F&period; Ivan and Burgers(线性基,离线)

    题目链接:http://codeforces.com/contest/1100/problem/F 题目大意:首先输入n,代表当前有n个数,然后再输入m,代表m次询问,每一次询问是询问区间[l,r], ...

  6. 【BZOJ-2460&amp&semi;3105】元素&amp&semi;新Nim游戏 动态维护线性基 &plus; 贪心

    3105: [cqoi2013]新Nim游戏 Time Limit: 10 Sec  Memory Limit: 128 MBSubmit: 839  Solved: 490[Submit][Stat ...

  7. 【题解】 bzoj3105&colon; &lbrack;cqoi2013&rsqb;新Nim游戏 (线性基&plus;贪心)

    bzoj3105,懒得复制 Solution: 首先你要有一个前置技能:如果每堆石子异或和为\(0\),则先手比输 这题我们怎么做呢,因为我们没人要先取掉几堆,为了赢对方一定会使剩下的异或和为\(0\ ...

  8. BZOJ&period;2460&period;&lbrack;BeiJing2011&rsqb;元素&lpar;线性基 贪心&rpar;

    题目链接 线性基:https://blog.csdn.net/qq_36056315/article/details/79819714. \(Description\) 求一组矿石,满足其下标异或和不 ...

  9. &lbrack;bzoj 2460&rsqb;线性基&plus;贪心&plus;证明过程

    题目链接:http://www.lydsy.com/JudgeOnline/problem.php?id=2460 网上很多题目都没说这个题目的证明,只说了贪心策略,我比较愚钝,在大神眼里的显然的策略 ...

随机推荐

  1. spring mvc 4识别浏览器(移动端) spring-mobile-device

    官方文档: http://projects.spring.io/spring-mobile/ 通过mvc.xml配置示例: http://blog.csdn.net/wuyt2008/article/ ...

  2. Docker网络详解

         当 Docker 启动时,会自动在主机上创建一个 docker0 虚拟网桥,实际上是 Linux 的一个 bridge,可以理解为一个软件交换机.它会在挂载到它的网口之间进行转发.      ...

  3. LoadRunner11录制APP脚本(2)

    通过安卓模拟器实现LoadRunner11录制APP脚本 http://www.51testing.com/html/24/15110424-3686857.html http://www.51tes ...

  4. SFTP和FTS协议的区别

    都是为FTP连接加密,协议非常相似.一个是借助SSL协议加密,一个时借助SSH协议加密.SSL是为HTTP/SMTP等加密设计的:SSH是为TELNET/FTP等加密.建立传输通道而设计的.其实SSH ...

  5. CSS 代码是什么&quest;(转)

    转自:http://www.divcss5.com/rumen/r95.shtml CSS 代码是什么,什么是CSS代码? 目录 什么是CSS css代码样子(图) 作用 相关扩展阅读 一.了解什么是 ...

  6. php zendstudio 常用的一些自定义模板标签

    <?php /** * xxx.php * ============================================== * Copy right 2013-2016 http: ...

  7. C&plus;&plus;异常处理 - 栈解旋,异常接口声明,异常类型和异常变量的生命周期

    栈解旋(unwinding) 异常被抛出后,从进入try块起,到异常被抛掷前,这期间在栈上的构造的所有对象,都会被自动析构.析构的顺序与构造的顺序相反.这一过程称为栈的解旋(unwinding). d ...

  8. go语言nsq源码解读一-基本介绍

    简单介绍一下nsq. 参考 http://feilong.me/2013/05/nsq-realtime-message-processing-system 的介绍:NSQ是由知名短链接服务商bitl ...

  9. Excel技巧--提取中文字串

    类似的,如果要提取上图第1列的商品,不要型号,如第2列. 可以考虑使用SEARCHB函数. searchb与search的区别,在于searchb函数以字节为单位搜索,search函数以字符为单位搜索 ...

  10. numexpr low version warning

    runing https://colab.research.google.com/notebooks/mlcc/first_steps_with_tensor_flow.ipynb?hl=zh-cn# ...