题目链接

挺好的博弈论题,这是一个跟官方题解不太一样的做法。

遇到这种组合游戏可以先考虑逆推胜负,把握一下规律,我们先从一个区间的胜负判断开始入手。

考察区间中最后一个数字的从属关系,如果它属于弈,因为 \(a_i>0\),如果前面传来的数字非空,则弈不用选择也可以获胜,否则只要选择当前的数即可。

如果它属于博,那么这个数得刚好为 0 或者等于 \(a_i\),博才有办法获胜,以此推广到最后有多个数属于博的情况:只有前面传来的数能被这些数异或表示(在它们的张成空间中),博才能获胜。

而这个规律推广到更前面的连续段也是适用的,即对于数列的任意一个前缀中属于弈的数的张成空间,都是对应后缀属于博的数的张成空间的子集时,博才能获胜。

找到了规律,我们接着来设计算法进行统计,一个维护这种“区间中的子区间”数量的常用方法是对一个端点进行扫描线,维护 \(S\) 数组表示以对应位置为开头或结尾,且以当前扫描到的这个数为另一个端点时,合法的区间数。

由于刚才我们是从右往左推理,我们也动态从右往左扫描,对于一个以弈拥有的数为右端点的区间贡献肯定为 \(r-l+1\),而另一种区间如果在扫描中遇到了一个节点,且它不在当前这个张成空间中,后面无论再怎么扫也一定无解,前面一定有解,也就是说这种胜负具有一定单调性。

这启发我们先预处理出每个弈拥有的点在右端点到哪才能被右边属于博的数异或表示出,记为 \(rpos_i\) 可以倒着扫描整个序列,贪心维护一个线性基,每次加数时淘汰时间比较老的节点,当加到 \(a_{i+1}\) 时就可以处理 \(a_i\) 的信息,对它用到的节点的时间戳取最大即可。

回到刚才的扫描线,由于“当前没有机会的右端点集合”只要一开始把所有都加进去就是只增不减,每次扫描到一个属于弈的数就把 \(i\) 到 \(rpos_i\) 中的所有数标记为合法,可以使用并查集或者 set 维护,之后开个树状数组维护答案,这里数点的方式有很多种,这里不再赘述。

#include<iostream>
#include<cstdio>
#include<algorithm>
#include<cstring>
#include<cmath>
#include<set>
#include<vector>
#define N 500005
#define M 1500006
#define ll long long
using namespace std;
typedef unsigned long long ul;
typedef unsigned int ui;
ui ans;
ui Ans;
ul Sd,Cnt;
ul cnt,n,q,tp,a[N];
ul Rd(){Sd^=Sd<<19,Sd^=Sd>>12,Sd^=Sd<<29;return Sd^=++Cnt;}
char s[N];
void GetA(ul &a){a=Rd()%((1ll<<60)-2)+1;}
void GetLR(ul &l,ul &r){
l=Rd()%n+1,r=Rd()%n+1;
if(l>r)swap(l,r);
}
ll read(){
ll x=0;char ch=getchar();
while(ch<'0' || ch>'9')ch=getchar();
while(ch>='0' && ch<='9'){
x=(x<<1)+(x<<3)+(ch^48);
ch=getchar();
}
return x;
}
//-----------------------------------------------
struct que{
ul l,r;
}Q[M];
void init(){
n=read();q=read();tp=read();
char ch=getchar();
while(ch<'0' || ch>'1')ch=getchar();
while(ch>='0' && ch<='1'){s[++cnt]=ch;ch=getchar();}
if(tp){
Sd=tp,Cnt=0;
for(ll i=1;i<=n;++i)GetA(a[i]);
for(ll qi=1;qi<=q;++qi){GetLR(Q[qi].l,Q[qi].r);}
}
else{
for(ll i=1;i<=n;i++)a[i]=read();
for(ll i=1;i<=q;i++){Q[i].l=read();Q[i].r=read();}
}
}
ll rpos[N];
vector<que> P[N];
ll bas[65],ti[65];
void insert(ll x,ll t){
for(ll i=61;i>=0;i--){
if(x&(1ll<<i)){
if(!bas[i]){bas[i]=x;ti[i]=t;return;}
else if(t<ti[i]){swap(x,bas[i]);swap(t,ti[i]);x^=bas[i];}
else x^=bas[i];
}
}
}
ll jud(ll x){
ll ans=0;
for(ll i=61;i>=0;i--){
if(x&(1ll<<i)){
x^=bas[i];ans=max(ans,ti[i]);
}
}
if(x) return n+1;
else return ans;
}
int ask_b(ll x,ll t){
ll ans=0;
for(ll i=61;i>=0;i--){
if(x&(1ll<<i) && t<=ti[i]){
x^=bas[i];ans=max(ans,ti[i]);
}
else if(x&(1ll<<i))return false;
}
}
set<ll> S;
ll F[N],G[N];
ll lowbit(ll x){
return x&(-x);
}
void add(ll x,ll val,ll *K){
while(x<=n){
K[x]+=val;
x+=lowbit(x);
}
}
ll ask(ll x,ll *K){
ll ans=0;
while(x){
ans+=K[x];
x-=lowbit(x);
}
return ans;
}
ui an[N];
int main(){
init();ll pos=1;
rpos[n]=n+1;
for(ll i=n;i>=1;i--){
if(s[i]=='0')insert(a[i],i);
else rpos[i]=jud(a[i]);
}
for(ll i=1;i<=n;i++)if(s[i]=='0'){S.insert(i);add(i,i,F);add(i,1,G);}
for(ll i=1;i<=q;i++)P[Q[i].l].push_back((que){i,Q[i].r});
for(ll i=n;i>=1;i--){
if(s[i]=='1'){
set<ll>::iterator itt=S.lower_bound(i);
set<ll>::iterator it=itt;
for(;itt!=S.end() && *itt<rpos[i];itt++){
add(*itt,-*itt,F);add(*itt,-1,G);
add(*itt,*itt-i,F);
}
S.erase(it,itt);
}
for(ll j=0;j<P[i].size();j++){
ll r=P[i][j].r;
ans=0;ans=ask(r,F)-ask(i-1,F)-(ask(r,G)-ask(i-1,G))*(i-1);
ans=(ll)(r-i+2)*(r-i+1)/2-ans;
if(!tp)an[P[i][j].l]=(ans%(1ll<<32));
ans%=(1ll<<32);
ans=(ans*P[i][j].l)%(1ll<<32);
ui tmp=ans;
Ans^=tmp;
}
}
if(tp)cout<<Ans;
else for(ll i=1;i<=q;i++)cout<<an[i]<<endl;
}

P9580 「Cfz Round 1」Wqs Game 题解的更多相关文章

  1. LibreOJ #525. 「LibreOJ β Round #4」多项式

    二次联通门 : LibreOJ #525. 「LibreOJ β Round #4」多项式 官方题解 : /* LibreOJ #525. 「LibreOJ β Round #4」多项式 由于会有多种 ...

  2. loj #547. 「LibreOJ β Round #7」匹配字符串

    #547. 「LibreOJ β Round #7」匹配字符串   题目描述 对于一个 01 串(即由字符 0 和 1 组成的字符串)sss,我们称 sss 合法,当且仅当串 sss 的任意一个长度为 ...

  3. [LOJ#531]「LibreOJ β Round #5」游戏

    [LOJ#531]「LibreOJ β Round #5」游戏 试题描述 LCR 三分钟就解决了问题,她自信地输入了结果-- > -- 正在检查程序 -- > -- 检查通过,正在评估智商 ...

  4. [LOJ#530]「LibreOJ β Round #5」最小倍数

    [LOJ#530]「LibreOJ β Round #5」最小倍数 试题描述 第二天,LCR 终于启动了备份存储器,准备上传数据时,却没有找到熟悉的文件资源,取而代之的是而屏幕上显示的一段话: 您的文 ...

  5. [LOJ#516]「LibreOJ β Round #2」DP 一般看规律

    [LOJ#516]「LibreOJ β Round #2」DP 一般看规律 试题描述 给定一个长度为 \(n\) 的序列 \(a\),一共有 \(m\) 个操作. 每次操作的内容为:给定 \(x,y\ ...

  6. [LOJ#515]「LibreOJ β Round #2」贪心只能过样例

    [LOJ#515]「LibreOJ β Round #2」贪心只能过样例 试题描述 一共有 \(n\) 个数,第 \(i\) 个数 \(x_i\) 可以取 \([a_i , b_i]\) 中任意值. ...

  7. [LOJ#525]「LibreOJ β Round #4」多项式

    [LOJ#525]「LibreOJ β Round #4」多项式 试题描述 给定一个正整数 k,你需要寻找一个系数均为 0 到 k−1 之间的非零多项式 f(x),满足对于任意整数 x 均有 f(x) ...

  8. [LOJ#526]「LibreOJ β Round #4」子集

    [LOJ#526]「LibreOJ β Round #4」子集 试题描述 qmqmqm有一个长为 n 的数列 a1,a2,……,an,你需要选择集合{1,2,……,n}的一个子集,使得这个子集中任意两 ...

  9. [LOJ#522]「LibreOJ β Round #3」绯色 IOI(危机)

    [LOJ#522]「LibreOJ β Round #3」绯色 IOI(危机) 试题描述 IOI 的比赛开始了.Jsp 和 Rlc 坐在一个角落,这时他们听到了一个异样的声音 …… 接着他们发现自己收 ...

  10. LibreOJ #517. 「LibreOJ β Round #2」计算几何瞎暴力

    二次联通门 : LibreOJ #517. 「LibreOJ β Round #2」计算几何瞎暴力 /* LibreOJ #517. 「LibreOJ β Round #2」计算几何瞎暴力 叫做计算几 ...

随机推荐

  1. 【SpringCloud】 Eureka

    Eureka 父工程 核心依赖 <?xml version="1.0" encoding="UTF-8"?> <project xmlns=& ...

  2. 【Docker】部署Tomcat

    搜索镜像 $ docker search 镜像名称:镜像TAG # 如: 没有加TAG,表示默认搜索的是最新版本的tomcat镜像 $ docker search tomcat # 如:搜索 tomc ...

  3. 前端Vuer,请收好这份《Vue组件单元测试》宝典,给自己多一些安全感

    大家好,我是 Kagol. 作为一名前端,在做业务开发的过程中,你是否曾经: 因为担心上线之后出bug,而反复手工验证自己负责的模块 不敢修改现有的"屎山"(别人写的或者是自己1年 ...

  4. BUUCTF-MISC-九连环(steghide隐写+伪加密)

    开局一张图 丢入winhex里,尾部有其他文件名 kali中binwalk -e x.jpg 得到的zip中 图片打不开 丢入winhex,发现jpg处是伪加密 修改之后,解压可打开图片,另一个压缩包 ...

  5. 从TL、ITL到TT

    1.概述 ThreadLocal(TL)是Java中一种线程局部变量实现机制,他为每个线程提供一个单独的变量副本,保证多线程场景下,变量的线程安全.经常用于代替参数的显式传递. Inheritable ...

  6. MyBatis(log4j)

    log4j介绍 Log4j是Apache的一个开源项目,通过使用Log4j,我们可以控制日志信息输送的目的地是控制台.文件.GUI组件,甚至是套接口服务器.NT的事件记录器.UNIX Syslog守护 ...

  7. debezium之mysql配置

    实验环境 全部部署于本地虚拟机 1 mysql 参考 官方文档 和 根据官方示例镜像(debezium/example-mysql,mysql版本为8.0.32) 1.1 创建用户 官方镜像里一共有三 ...

  8. EF6连接oracle

    最近项目用到oracle,一直使用sql server,ef很方便连接mssql,但是连接oracle的方法网上很多文章,尝试很多次终于搞定,dbfirst或者codefirst也是可以的. 安装OD ...

  9. 【WebGL系列-01】获取WebGL上下文

    获取WebGL上下文 获取上下文 WebGL上下文是从<canvas>标签中获取到的,通过canvas对象的getContext()函数可以获取WebGLRenderingContext. ...

  10. Git:多人写作时,如何保证代码一致性

    解决方案 git add . git commit -m "message" git pull origin develop # 拉取并合并dev分支上的代码 git push