简要题意

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

你需要维护一个数据结构,支持区间异或和区间求与 \(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. XAF新手入门 - 类型子系统(Types Info Subsystem)

    类型子系统概述 类型子系统是XAF的核心概念,但我们平时却很少关注它,它集中存储了模块中的类型,它是生成应用程序模型(Application Model)的基础,它与XAF中其它的概念都有所关联,了解 ...

  2. 三、redis环境安装

    三.redis环境安装 3.1.下载和安装 下载地址:https://github.com/tporadowski/redis/releases   使用以下命令启动redis服务端 redis-se ...

  3. Python基础部分:7、 垃圾回收机制和流程控制

    目录 一.垃圾回收机制 1.引用计数 2.标记清除 3.分类代收 二.流程控制 1.理论 2.必备知识 3.分支结构 4.循环结构 一.垃圾回收机制 垃圾回收机制,简称GC,是python解释器自带的 ...

  4. java学习之spring基础

    0x00前言 spring框架应用的是ioc模式,ioc模式是指控制反转模式,本质是你不去创建对象让spring框架给你创建对象你去使用对象.多种开发模式通过配置文件和注解的方式去开发的都很值得去学习 ...

  5. phpword 模板文件导出word到服务器 并浏览器下载

    模板文件填充 然后生成新文件 //调用PHPwordrequire_once(ROOTPATH . "inc/vendor/autoload.php"); $phpWord = n ...

  6. swap,传参实质

    void swap(int a,int b){ int s=a; a=b; b=s; } int main(){ int x=1,y=2; swap(x,y); } 上面的函数并不能实现交换,因为传参 ...

  7. 重学c#系列——委托和匿名函数[二十五]

    前言 简单介绍一下什么是委托. 正文 以前也写过委托,这次算是重新归档,和新的补充吧. https://www.cnblogs.com/aoximin/p/13940125.html 有些人说委托是函 ...

  8. 基于.NetCore开发博客项目 StarBlog - (20) 图片显示优化

    前言 我的服务器带宽比较高,博客部署在上面访问的时候几乎没感觉有加载延迟,就没做图片这块的优化,不过最近有小伙伴说博客的图片加载比较慢,那就来把图片优化完善一下吧~ 目前有两个地方需要完善 图片瀑布流 ...

  9. 兼容IE全版本及所有市面浏览器的网页变黑白处理方式

    大家应该有发现最近几天不少网站变成了黑白色,在哀悼日时,很多网站都需要全站变成黑白配色,今天对这个实现的技术做了一些探索性了解,在此进行一个记录分享. 使用的样式部分:下面的css部分想必大家应该都可 ...

  10. C++使用ODBC连接数据库遇到的问题

    C++使用ODBC连接数据库遇到的问题 1.SQL语句中包含中文无法正常执行的问题 2.字符与宽字符相互转化的问题 C++使用ODBC连接数据库遇到的问题 1.SQL语句中包含中文无法正常执行的问题 ...