简要题意

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

你需要维护一个数据结构,支持区间异或和区间求与 \(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. .NET 6学习笔记(4)——如何在.NET 6的Desktop App中使用Windows Runtime API

    Windows Runtime API是当初某软为了区别Win32 API,力挺UWP而创建的另一套Windows 10专用的API集合.后来因为一些原因,UWP没火.为了不埋没很有价值的Window ...

  2. MySQL数据库的性能分析 ---图书《软件性能测试分析与调优实践之路》-手稿节选

    1  .MySQL数据库的性能监控 1.1.如何查看MySQL数据库的连接数 连接数是指用户已经创建多少个连接,也就是MySQL中通过执行 SHOW  PROCESSLIST命令输出结果中运行着的线程 ...

  3. vscode代码部署

    前言 在本地环境中开发代码时,经常需要将代码上传到服务器环境中,在环境中构建并调试程序.如果手动使用scp.ftp等文件传输程序将代码上传至环境,一次两次还好,反复多次操作则有些繁琐. 为了方便进行本 ...

  4. EntityFramework介绍

    首先我们说明一下ORM是什么. 微软官方提供的ORM工具,ORM让开发人员节省数据库访问的代码时间,将更多的时间放到业务逻辑层代码上.开发人员使用Linq语言,对数据库操作如同操作Object对象 一 ...

  5. day03-实现02

    实现02 3.实现任务阶段3-处理Servlet02 3.3Servlet规范设计 3.3.1MyServlet 该类模仿Servlet接口,为了简化,只声明了三个方法:init(),service( ...

  6. JUC学习笔记——并发工具线程池

    JUC学习笔记--并发工具线程池 在本系列内容中我们会对JUC做一个系统的学习,本片将会介绍JUC的并发工具线程池 我们会分为以下几部分进行介绍: 线程池介绍 自定义线程池 模式之Worker Thr ...

  7. 基于python的数学建模---非线性规划

    凸函数的非线性规划 minimize 求解的是局部最优解 简单的函数,无所谓 复杂的函数 初始值的设定很重要 scipy.optimize.minimize(fun,x0,args=(),method ...

  8. Java开发学习(四十二)----MyBatisPlus查询语句之条件查询

    一.条件查询的类 MyBatisPlus将书写复杂的SQL查询条件进行了封装,使用编程的形式完成查询条件的组合. 这个我们在前面都有见过,比如查询所有和分页查询的时候,都有看到过一个Wrapper类, ...

  9. vscode配置remote ssh

    前言 简单用vscode配置 remote ssh可以实现,通过ssh 在线使用vscode编辑文件,很方便,也遇到一些坑. 安装插件 设置界面 右键最左边tab栏: 勾选 远程资源管理器 添加远程服 ...

  10. ArcGIS 通过字段计算 设置顺序编码

    地块编号="前缀" & left("0000",4-len( [FID]+1)) & ([FID] +1)