@bzoj - 4035@ [HAOI2015]数组游戏
@description@
有一个长度为N的数组,甲乙两人在上面进行这样一个游戏:
首先,数组上有一些格子是白的,有一些是黑的。然后两人轮流进行操作。每次操作选择一个白色的格子,假设它的下标为x。接着,选择一个大小在1~n/x之间的整数k,然后将下标为x、2x、...、kx的格子都进行颜色翻转。不能操作的人输。
现在甲(先手)有一些询问。每次他会给你一个数组的初始状态,你要求出对于这种初始状态他是否有必胜策略。
@solution@
考虑一个巧妙(至少我觉得很巧妙)的转化:我们把白色格子看作初始有一个棋子,颜色翻转变成直接在格子上放棋子。
当一个格子有多于 1 个棋子存在时,先手对该格子操作,后手可以模仿相同的操作。因此并不影响。
这样子转化的好处是:我们把每个白格子独立出来,变成互不干涉的组合游戏。因此就可以使用 sg 函数来刻画了。
可以列出转移 \(sg(x) = mex(0, sg(2x), sg(2x)\oplus sg(3x), \dots)\)。
然而 n 很大,考虑怎么优化。
注意当 x > n/2 时 sg(x) 相同,继续算发现 n/2 >= x > n/3 时 sg(x) 也相同,因此不难猜测到 \(sg(x) = f(\lfloor\frac{n}{x}\rfloor)\)。证明根据 sg 的转移式易证。
由 \(sg(x) = mex(0, sg(2x), sg(2x)\oplus sg(3x), \dots)\) 可得 f 的转移:
f(p) = mex(0, f(\lfloor\frac{p}{2}\rfloor), f(\lfloor\frac{p}{2}\rfloor)\oplus f(\lfloor\frac{p}{3}\rfloor), \dots)
\]
分块转移即可。存储 f 用类似杜教筛的方法即可。
时间复杂度 \(\sum_{i=1}^{\sqrt{n}}(\sqrt{\frac{n}{i}} + \sqrt{i})\)。积分拟合一下大概是 \(O(n^{\frac{3}{4}})\)。
@accepted code@
#include <cmath>
#include <cstdio>
#include <algorithm>
using namespace std;
const int SQRT = 32000;
int N; int sg1[SQRT + 5], sg2[SQRT + 5];
int *sg(int x) {return x > SQRT ? sg1 + (N/x) : sg2 + x;}
int a[2*SQRT + 5], vis[2*SQRT + 5], cnt;
void get() {
	for(int i=1;i<=N;i=(N/(N/i))+1) a[++cnt] = N/i;
	for(int i=cnt;i>=1;i--) {
		int tmp = 0;
		for(int j=2;j<=a[i];) {
			int p = a[i] / j, k = a[i] / p, x = (*sg(p)) ^ tmp;
			vis[x] = i;
			if( (k - j + 1) & 1 )
				tmp = x;
			j = k + 1;
		}
		int ans = 1;
		while( vis[ans] == i ) ans++;
		(*sg(a[i])) = ans;
	}
}
int main() {
	scanf("%d", &N), get();
	int K; scanf("%d", &K);
	for(int i=1;i<=K;i++) {
		int W, ans = 0; scanf("%d", &W);
		for(int j=1;j<=W;j++) {
			int x; scanf("%d", &x);
			ans ^= (*sg(N/x));
		}
		puts(ans ? "Yes" : "No");
	}
}
@details@
能少用除法就少用除法,毕竟最慢运算符(好像不是,取模是最慢的),很可能(像我一样)被卡常。
@bzoj - 4035@ [HAOI2015]数组游戏的更多相关文章
- 【BZOJ 4035】 4035: [HAOI2015]数组游戏 (博弈)
		
4035: [HAOI2015]数组游戏 Time Limit: 15 Sec Memory Limit: 32 MBSubmit: 181 Solved: 89 Description 有一个长 ...
 - bzoj4035 [HAOI2015]数组游戏
		
这题显然把每个白格子看成一个子游戏 一个白格子$x$的$sg$值是$mex{[0,sg[2x],sg[2x] XOR sg[3x].....]}$ 打表发现一个数的$sg$值只和$n/x$有关,然后分 ...
 - [HAOI2015]数组游戏
		
题目大意: 有一排n个格子,每个格子上都有一个白子或黑子,在上面进行游戏,规则如下: 选择一个含白子的格子x,并选择一个数k,翻转x,2x,...,kx格子上的子. 不能操作者负. 思路: 将“某个格 ...
 - 【BZOJ4035】数组游戏(博弈论)
		
[BZOJ4035]数组游戏(博弈论) 题面 BZOJ 洛谷 题解 很明显是一个翻硬币游戏的变形,因此当前局面的\(SG\)函数值就是所有白格子单独存在的\(SG\)函数的异或和. 那么,对于每一个位 ...
 - bzoj 3991: [SDOI2015]寻宝游戏 虚树 set
		
目录 题目链接 题解 代码 题目链接 bzoj 3991: [SDOI2015]寻宝游戏 题解 发现每次答案就是把虚树上的路径*2 接在同一关键点上的点的dfs序是相邻的 那么用set动态维护dfs序 ...
 - bzoj 3232: 圈地游戏
		
bzoj 3232: 圈地游戏 01分数规划,就是你要最大化\(\frac{\sum A}{\sum B}\),就二分这个值,\(\frac{\sum A}{\sum B} \geq mid\) \( ...
 - BZOJ.4034 [HAOI2015]树上操作 ( 点权树链剖分 线段树 )
		
BZOJ.4034 [HAOI2015]树上操作 ( 点权树链剖分 线段树 ) 题意分析 有一棵点数为 N 的树,以点 1 为根,且树点有边权.然后有 M 个 操作,分为三种: 操作 1 :把某个节点 ...
 - [BZOJ 4820] [SDOI2017] 硬币游戏(高斯消元+概率论+字符串hash)
		
[BZOJ 4820] [SDOI2017] 硬币游戏(高斯消元+概率论+字符串hash) 题面 扔很多次硬币后,用H表示正面朝上,用T表示反面朝上,会得到一个硬币序列.比如HTT表示第一次正面朝上, ...
 - BZOJ 4033: [HAOI2015]树上染色题解
		
BZOJ 4033: [HAOI2015]树上染色题解(树形dp) 标签:题解 阅读体验:https://zybuluo.com/Junlier/note/1327400 原题地址: BZOJ 403 ...
 
随机推荐
- mysql小白系列_08 zabbix添加自定义监控项items和触发器
			
监控mysql存活 1.配置agent自定义参数 vi /usr/local/zabbix/etc/zabbix_agentd.conf Include=/usr/local/zabbix/etc/z ...
 - module.exports = $;  $ is not defined
			
https://blog.csdn.net/weixin_43945983/article/details/88294052 解决方案:安装依赖包 1.执行安装jquery依赖包命令 cnpm ins ...
 - centos 删除文件提示 Operation not permitted
			
如果文件上存在 i 标记,那肯定是删不掉的,同样这个文件也不能被编辑.可以进入 root 模式,去除这个标记: root@ubuntu:/home/barret/work# chattr -i 1.m ...
 - net core获取appsetting.json的另外一种思路(全局,实时变化无需重启项目)
			
最近在写net core的项目,在非controller和service里面需要用到appsetting.json文件里面的一些配置,查资料大概有几种思路: 注入,然后config.GetSectio ...
 - RESTful api 功能测试
			
0 为什么要写测试代码 代码写好了,如果能点或者能看,开发人员一般会自己点点或看看,如果没有发现问题就提交测试:更进一步,代码写好后,运行测试代码,通过后提交测试.将流程抽象下: 功能1编码-> ...
 - 漫谈碎片化学习(Fragmentation learning)
			
碎片化学习(Fragmentation Learning) 从一个互联网小段子讲起: 某天,美国情报部门FBI应奥巴马的要求,做相关汇报:“报告总统,经FBI分析,中国‘短信’中35%是节日祝福语,2 ...
 - Java IO(十四) CharArrayReader 和 CharArrayWriter
			
Java IO(十四) CharArrayReader 和 CharArrayWriter 一.介绍 CharArrayReader 和 CharArrayWriter 是字符数组输入流和字符数组输出 ...
 - This关键字练习
			
Account: package com.aff.ex; public class Account { private int id;// 账号 private double balance;// 余 ...
 - DataFrame的apply用法
			
DataFrame的apply方法: def cal_value_percent(row,total_value): row['new_column']=row[estimated_value_col ...
 - 【书签】stacking、blending
			
读懂stacking:模型融合Stacking详解/Stacking与Blending的区别 https://blog.csdn.net/u014114990/article/details/5081 ...