简要题意

其实我觉得这个部分可以不要,因为这道题的题面还是很清晰的。

你需要维护一个数据结构,支持区间异或和区间求与 \(v\) 的最大异或和。

思路

对于这种区间问题,最容易想到的就是 分块 线段树。

而对于复杂的异或问题,最容易想到的就是 01 Trie 线性基。

合在一起,就是线段是套线性基。(好像还用了顶针的手法)

做这道题之前建议先做 P4839 P哥的桶 那是这一道题的弱化版。

(现在默认大家已经做过P哥的桶了)

P哥的桶中,是单点修改,而这道题是区间修改,用传统的打tag是不方便维护的,我们可以用差分的思想。

Sol1大佬的文章已经有了正确性证明,我这里就口胡一下结论:

我们可以维护异或差分数组 \(b\),然后每次询问 \([l,r]\) 就在线段树上询问 \([l+1,r]\),再插入 \(a_l\) 即可。

然后就是一个差分的过程,随便找一个数据结构维护一下即可。(这里是树状数组)

细节

(由于没特判 \(l=r\) \(45\) 分挂了很久)

注意要特判 \(l=r\),如果 \(l=r\),那么就可以分类讨论一下,答案就是 \(\max\{\operatorname{BIT.query}(l) \operatorname{xor} v,v\}\)。

下面代码的注释也有说明(虽然是蹩脚的英文的)

代码

(纪念,这是我A的第二道Ynoi,也是第一篇试图提交题解的Ynoi)

里面有一份线性基模板,如果您觉得我的代码好,可以参考。

不用卡常,好耶!

#include <bits/stdc++.h>
#define int long long
using namespace std; namespace Basis{
const int MAX_BIT = 30;
struct Basis{
int p[MAX_BIT+5];
int _how_many_numbers_can_xor;
void clear(){
memset(p,0,sizeof(p));
_how_many_numbers_can_xor=0;
}
Basis(){
clear();
}
void insert(int x){
for(int i=MAX_BIT;i>=0;i--){
if(!(x&(1ll<<i)))continue;
if(!p[i]){
p[i]=x;
_how_many_numbers_can_xor++;
break;
}
x^=p[i];
}
}
int max_xor(int init=0){
int ans=init;
for(int i=MAX_BIT;i>=0;i--){
if((ans^p[i])>ans){
ans^=p[i];
}
}
return ans;
}
bool can_be_xor(int x){
for(int i=MAX_BIT;i>=0;i--){
if(x&(1ll<<i))x^=p[i];
}
return x==0;
}
int numbers_can_xor(){
return (1ll<<_how_many_numbers_can_xor);
}
void expand(Basis &x){
for(int i=MAX_BIT;i>=0;i--){
if(x.p[i]){
insert(x.p[i]);
}
}
}
}; Basis merge(Basis x,Basis y){
for(int i=MAX_BIT;i>=0;i--){
if(y.p[i])x.insert(y.p[i]);
}
return x;
}
} int n,m;
int a[50005]; namespace sgt{
Basis::Basis t[200005]; void pushup(int i){
t[i]=Basis::merge(t[i<<1],t[i<<1|1]);
}
void build(int i,int l,int r){
for(int ii=l;ii<=r;ii++){
t[i].insert(a[ii]);
}
if(l==r){
return;
}
int mid=(l+r)>>1;
build(i<<1,l,mid);
build(i<<1|1,mid+1,r);
pushup(i);
}
void update(int x,int v,int i,int l,int r){
if(l==r){
t[i].clear();
a[l]^=v;
t[i].insert(a[l]);
return;
}
int mid=(l+r)>>1;
if(x<=mid){
update(x,v,i<<1,l,mid);
}
else{
update(x,v,i<<1|1,mid+1,r);
}
pushup(i);
}
Basis::Basis query(int ql,int qr,int i,int l,int r){
if(ql<=l&&r<=qr){
return t[i];
}
Basis::Basis result;
int mid=(l+r)>>1;
if(ql<=mid){
result=merge(result,query(ql,qr,i<<1,l,mid));
}
if(qr>mid){
result=merge(result,query(ql,qr,i<<1|1,mid+1,r));
}
return result;
}
} namespace BIT{
int t[50005];
void clear(){
memset(t,0,sizeof(t));
}
inline int lowbit(int x){
return x&(-x);
}
void update(int p,int v){
while(p<=n){
t[p]^=v;
p+=lowbit(p);
}
}
int query(int p){
int ret=0;
while(p){
ret^=t[p];
p-=lowbit(p);
}
return ret;
}
} void getdiff(){
for(int i=n;i>1;i--){
a[i]^=a[i-1];
}
} signed main(){
ios::sync_with_stdio(false);
cin.tie(nullptr);
cout.tie(nullptr);
cin>>n>>m;
for(int i=1;i<=n;i++){
cin>>a[i];
}
BIT::clear();
getdiff();
for(int i=1;i<=n;i++){
BIT::update(i,a[i]);
}
sgt::build(1,1,n);
while(m--){
int op,l,r,v;
cin>>op>>l>>r>>v;
if(op==1){
BIT::update(l,v);
sgt::update(l,v,1,1,n);
if((r+1)<=n){
BIT::update(r+1,v);
sgt::update(r+1,v,1,1,n);
}
}
else{
int lft=BIT::query(l);
if(l==r){
// If l equals r
// Then there are only two possibilities:LFT xor v | v
cout<<max(v,lft^v)<<'\n';
continue;
}
Basis::Basis result=sgt::query(l+1,r,1,1,n);
result.insert(lft);
cout<<(result.max_xor(v))<<'\n';
}
}
return 0;
}

题解 P5607 [Ynoi2013] 无力回天 NOI2017的更多相关文章

  1. 洛谷 P5607 [Ynoi2013] 无力回天 NOI2017

    人生第一道Ynoi,开心 Description https://www.luogu.com.cn/problem/P5607 Solution 拿到这个题,看了一下,发现询问要求最大异或和,怎么办? ...

  2. P5607-[Ynoi2013]无力回天NOI2017【线性基,线段树,树状数组】

    正题 题目链接:https://www.luogu.com.cn/problem/P5607 题目大意 \(n\)个数字的序列,\(m\)次操作 区间\([l,r]\)异或上一个值\(v\) 询问区间 ...

  3. 线段树套线性基——题解P4839 P哥的桶

    文章历史 2022-08-03: 文章初稿,由于对算法介绍过于少而被管理员打回重造. 2020-08-06:将算法介绍进行扩写,并删除了一些可有可无的内容或玩梗内容. 管理员审核题解辛苦了. 简要题意 ...

  4. BZOJ4946 & 洛谷3826 & UOJ318:[NOI2017]蔬菜——题解

    https://www.lydsy.com/JudgeOnline/problem.php?id=4946 https://www.luogu.org/problemnew/show/P3826 ht ...

  5. [UOJ317]【NOI2017】游戏 题解

    题意 ​ 小 L 计划进行 \(n\) 场游戏,每场游戏使用一张地图,小 L 会选择一辆车在该地图上完成游戏. ​ 小 L 的赛车有三辆,分别用大写字母 A.B.C 表示.地图一共有四种,分别用小写字 ...

  6. BZOJ4943 & 洛谷3823 & UOJ315:[NOI2017]蚯蚓排队——题解

    https://www.lydsy.com/JudgeOnline/problem.php?id=4943 http://uoj.ac/problem/315 https://www.luogu.or ...

  7. noi2017 day1 题解

    d1t1 sol1:用线段树维护区间是否全0/全1,叶子上压位维护对应位置的数位,加法首先对叶子加,如需进位则向右找到第一个不是全1的叶子+1,中间部分全1部分打上反转标记,减法同理. #includ ...

  8. BZOJ4945 & 洛谷3825 & UOJ317:[NOI2017]游戏——题解

    https://www.lydsy.com/JudgeOnline/problem.php?id=4945 https://www.luogu.org/problemnew/show/P3825 ht ...

  9. BZOJ4942 & UOJ314:[NOI2017]整数——题解

    https://www.lydsy.com/JudgeOnline/problem.php?id=4942 http://uoj.ac/problem/314 https://www.luogu.or ...

随机推荐

  1. Ubuntu安装错误 server64 busybox-initramfs安装失败

    因为想试试在Linux系统上爆破 所以安装了一下Ubuntu.第一次安装包报了个server64 busybox-initramfs.在安装系统那边.我一直还以为是我磁盘分配错了. 后来在网上找了资料 ...

  2. java中获取当前执行线程的名称

    Thread.currentThread().getName()

  3. 13.内建函数eval()

      eval函数 eval()函数十分强大 -- 将字符串当成有效的表达式来求值并返回计算结果 例如下图,eval会将字符串的引号去掉并且计算返回结果  

  4. 解决@Url.Action("Action", "Controller",new {p1=v1,p2=v2 })的传参问题

    1.首先@Url.Action("Action", "Controller",new {p1=v1,p2=v2 })后面的model参数不可以直接用变量 需要先 ...

  5. Windows7下驱动开发与调试体系构建——4.在x64下使用汇编代码(x86下的_asm)

    目录/参考资料:https://www.cnblogs.com/railgunRG/p/14412321.html asm文件设置 在vs x64中无法使用_asm关键字,需要使用.asm文件. 按第 ...

  6. Python 多重继承时metaclass conflict问题解决与原理探究

    背景 最近有一个需求需要自定义一个多继承abc.ABC与django.contrib.admin.ModelAdmin两个父类的抽象子类,方便不同模块复用大部分代码,同时强制必须实现所有抽象方法,没想 ...

  7. 企业级自定义表单引擎解决方案(十六)--Excel导入导出

    Excel对于后端管理系统来说,永远都是绕不开的话题,开发Excel导入导出功能往往都比较麻烦,因为涉及到Excel导入模板制作.Excel表格数据与系统数据库表字段映射.Excel导入数据验证.验证 ...

  8. vue-项目的整体增删改查

    Dept: package com.example.demo.gs; public class Dept { private int id; private String name; private ...

  9. jquery datatable 粗犷

    需要学习: HTML.CSS.Javascript Bootstrap: 基于以上三个的一个框架 jQuery:一个 JavaScript 库. jQuery 极大地简化了 JavaScript 编程 ...

  10. 2022-11-01 Acwing每日一题

    第k个数 给定一个长度为 n 的整数数列,以及一个整数 k,请用快速选择算法求出数列从小到大排序后的第 k 个数. 输入格式 第一行包含两个整数 n 和 k. 第二行包含 n 个整数(所有整数均在 1 ...