HDU 6610 Game — 2019第三场杭电多校 1008题
@
题意
大概说一下我理解的题意。。。
链接:here
你有\(n\)堆石子,每堆石子有\(a_i\)个石子。游戏规则:\(Alice\)先选择一个大范围\([L,R]\)区间内的石子,\(Bob\)选择一个子区间\([l,r]\)内的石子最终进行游戏。每次至少取走某一堆的一个石子,至多全部取走,无法移动石子者输。\(Alice\)先手,双方足够聪明。问对\(Alice\)的每次选择\([L_i,R_i]\),\(Bob\)有多少种选择能让\(Alice\)必胜。
还有修改操作,即交换相邻的两堆石子。
思路
Nim博弈,区间异或和,带修改莫队
- \(nim\)博弈结论,区间异或和为\(0\),则先手必败。
- 问题转换为维护区间异或和为\(0\)的对数,对序列做前缀异或和,莫队维护前缀异或和出现的次数,基本操作。
- 带修改?交换相邻两堆石子,只会对左堆石子的前缀异或和造成影响,即单点修改。
- 参考bzoj_2120数颜色,加一个修改时间戳标记,每次询问前回到当时修改时间戳即可。
- 带修改莫队的块的大小为\(n^{\frac 23}\)较优。
写了很多\(bug\),嘤嘤嘤,太久没写过莫队, 代码里面一堆\(debug\)操作。。。汗
AC_Code
const int MXN = 1e5 + 5;
const int MXE = 3e6 + 6;
int n, m;
LL ANS[MXN], ans;
int bel[MXN];
struct lp {
int l, r, id, tim;
}cw[MXN];
struct lh {
int x, oldx, newx;
}tim[MXN];
int vis[MXE], ar[MXN], res[MXN], ret[MXN];
LL L, R;
bool cmp(const lp&a, const lp&b) {
if(bel[a.l] != bel[b.l]) return a.l < b.l;
if(bel[a.r] != bel[b.r]) {
if(bel[a.l] & 1) return a.r < b.r;
return a.r > b.r;
}
return a.tim < b.tim;
}
inline void up(int p) {
++ vis[res[p]];
ans += vis[res[p]] - 1;
}
inline void down(int p) {
-- vis[res[p]];
ans -= vis[res[p]];
}
inline void upT(int t) {
res[tim[t].x] = tim[t].newx;
if(L <= tim[t].x && tim[t].x <= R) {
++ vis[tim[t].newx];
// debug(tim[t].oldx, tim[t].newx)
ans += vis[tim[t].newx] - 1;
-- vis[tim[t].oldx];
ans -= vis[tim[t].oldx];
}
}
inline void downT(int t) {
res[tim[t].x] = tim[t].oldx;
if(L <= tim[t].x && tim[t].x <= R) {
-- vis[tim[t].newx];
ans -= vis[tim[t].newx];
++ vis[tim[t].oldx];
ans += vis[tim[t].oldx] - 1;
}
}
int main() {
#ifndef ONLINE_JUDGE
freopen("/home/cwolf9/CLionProjects/ccc/in.txt", "r", stdin);
//freopen("/home/cwolf9/CLionProjects/ccc/out.txt", "w", stdout);
#endif
// int Tim = read();
while(~scanf("%d%d", &n, &m)) {
int block = (int)pow(n, 2. / 3);//标程和10取了max,不知道为啥
int Max = 0;
for(int i = 1; i <= n; ++i) {
ar[i] = read();
bel[i] = (i-1)/block + 1;
res[i] = res[i-1] ^ ar[i];
ret[i] = res[i];
Max = big(Max, ar[i], res[i]);
}
int change = 0, cnt1 = 1, cnt2 = 0;
for(int i = 1, a; i <= m; ++i) {
a = read();
if(a == 1) {
cw[cnt1].l = read(), cw[cnt1].r = read();
cw[cnt1].tim = change;
cw[cnt1].id = cnt1;
++ cnt1;
}else {
tim[cnt2].x = read();
tim[cnt2].oldx = res[tim[cnt2].x];
tim[cnt2].newx = (res[tim[cnt2].x + 1] ^ ar[tim[cnt2].x]);
res[tim[cnt2].x] = tim[cnt2].newx;
Max = big(Max, tim[cnt2].newx);
swap(ar[tim[cnt2].x], ar[tim[cnt2].x + 1]);
// debug(cnt2, tim[cnt2].oldx, tim[cnt2].newx)
++ change; ++ cnt2;
}
}
for(int i = 0; i <= Max; ++i) vis[i] = 0;
sort(cw + 1, cw + cnt1, cmp);
L = R = ans = 0;
up(0);
for(int i = 0; i <= n; ++i) res[i] = ret[i];
for(int i = 1, t = 0, f = 1; i < cnt1; ++i) {
while(R < cw[i].r) up(++ R);
while(R > cw[i].r) down(R --);
while(L < cw[i].l - 1) down(L ++);
while(L >= cw[i].l) up(-- L);
// for(int j = 0; j <= 7; ++j) printf("%d ", res[j]); printf("\n");
// for(int j = 0; j <= 16; ++j) printf("%d ", vis[j]); printf("\n");
for(;t < cw[i].tim; ++ t) upT(t);
for(;t > cw[i].tim; -- t) downT(t-1);
// for(int j = 0; j <= 7; ++j) printf("%d ", res[j]); printf("\n");
// for(int j = 0; j <= 16; ++j) printf("%d ", vis[j]); printf("\n");
// printf("*%lld %d %d %d\n", ans, L, R, cw[i].tim);
ANS[cw[i].id] = (R - L + 1)*(R - L)/2 - ans;
}
for(int i = 1; i < cnt1; ++i) write(ANS[i]);
}
#ifndef ONLINE_JUDGE
cout << "time cost:" << 1.0*clock()/CLOCKS_PER_SEC << "ms" << endl;
#endif
return 0;
}
HDU 6610 Game — 2019第三场杭电多校 1008题的更多相关文章
- HDU 5762 Teacher Bo (鸽笼原理) 2016杭电多校联合第三场
题目:传送门. 题意:平面上有n个点,问是否存在四个点 (A,B,C,D)(A<B,C<D,A≠CorB≠D)使得AB的横纵坐标差的绝对值的和等于CD的横纵坐标差的绝对值的和,n<1 ...
- HDU 5795 A Simple Nim (博弈) ---2016杭电多校联合第六场
A Simple Nim Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 65536/65536 K (Java/Others)Tota ...
- HDU 5744 Keep On Movin (贪心) 2016杭电多校联合第二场
题目:传送门. 如果每个字符出现次数都是偶数, 那么答案显然就是所有数的和. 对于奇数部分, 显然需要把其他字符均匀分配给这写奇数字符. 随便计算下就好了. #include <iostream ...
- HDU6655 Just Repeat(2019杭电多校J题)
原题链接:http://acm.hdu.edu.cn/showproblem.php?pid=6655 简单博弈问题,A,B手里各有n,m张牌,牌有颜色,两人轮流出牌(A先出),一个人只能打出对放未打 ...
- 可持久化线段树的学习(区间第k大和查询历史版本的数据)(杭电多校赛第二场1011)
以前我们学习了线段树可以知道,线段树的每一个节点都储存的是一段区间,所以线段树可以做简单的区间查询,更改等简单的操作. 而后面再做有些题目,就可能会碰到一种回退的操作.这里的回退是指回到未做各种操作之 ...
- 2018 Multi-University Training Contest 9 杭电多校第九场 (有坑待补)
咕咕咕了太久 多校博客直接从第三场跳到了第九场orz 见谅见谅(会补的!) 明明最后看下来是dp场 但是硬生生被我们做成了组合数专场…… 听说jls把我们用组合数做的题都用dp来了遍 这里只放了用组 ...
- 2018 Multi-University Training Contest 1 杭电多校第一场
抱着可能杭电的多校1比牛客的多校1更恐怖的想法 看到三道签到题 幸福的都快哭出来了好吗 1001 Maximum Multiple(hdoj 6298) 链接:http://acm.hdu.edu. ...
- HDU 5754 Life Winner Bo (各种博弈) 2016杭电多校联合第三场
题目:传送门 题意:一个国际象棋棋盘,有四种棋子,从(n,m)走到(1,1),走到(1,1)的人赢,先手赢输出B,后手赢输出G,平局输出D. 题解:先把从(n,m)走到(1,1)看做是从(1,1)走到 ...
- hdu 5319 Painter(杭电多校赛第三场)
题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=5319 Painter Time Limit: 2000/1000 MS (Java/Others) ...
随机推荐
- Php单元测试 phpunit & codecept
phpunit: Windows版本 整体上说,在 Windows 下安装 PHAR 和手工在 Windows 下安装 Composer 是一样的过程: 下载链接:http://pan.baidu.c ...
- 【Web API]无法添加AttributeRoutes的解决方案
1.按照微软官方文档,如果要使用AttributeRoutes,需要在APP_START里的WebApiConfig.cs的Register方法中添加一行:config.MapHttpAttribut ...
- 洛谷P4169 [Violet]天使玩偶/SJY摆棋子(CDQ分治)
[Violet]天使玩偶/SJY摆棋子 题目传送门 解题思路 用CDQ分治开了氧气跑过. 将输入给的顺序作为第一维的时间,x为第二维,y为第三维.对于距离一个询问(ax,ay),将询问分为四块,左上, ...
- MySQL开启SSL认证,以及简单优化
1.1 MySQL开启SSL认证 #生成一个 CA 私钥 [root@db01 ssl]# openssl genrsa 2048 > ca-key.pem Generating RSA pri ...
- [CF895E] Eyes Closed(线段树,期望)
Desctiption 传送门:Portal 大致题意: 给你一个序列, 支持两种操作: 1 l1 r1 l2 y2 在\([l1, r1]\)随机选择一个数a, \([l2, r2]\) 内随机选择 ...
- Spring Cloud Gateway注册到服务器中心(Consul)
Spring Cloud Gateway注册到服务器中心(Consul) 准备环境 启动Consul(./consul agent -dev)作为服务中心,默认是8500端口,然后启动spring-c ...
- SpringCloud 使用Feign访问服务
Feign简介: 声明式的Rest WEB 服务的客户端, https://github.com/OpenFeign/feign.Spring Cloud 提供了Spring-cloud-start ...
- v8引擎的优化
1.编译优化 V8采用JIT即使编译技术. 例如JAVA是先编译成字节码,再由JVM编译成机器码,V8则没有中间的字节码,直接由源码生成语法树,然后编译成机器码. 2.隐藏类 当定义一个构造函数,使用 ...
- MVC的实体模型写在类库,为什么被其他类库调用时,用不了模型的表?
一,很简单,由于第一次添加实体模型时,VS会自动帮你添加引用System.Data.Entity到当前类库,如下图示: 二,而手动添加的类库并不存在这个引用,则及时你引用了当前的实体模型的类库,却使用 ...
- 图解SSH原理_20190613
SSH仅仅是一协议标准,其具体的实现有很多,既有开源实现的OpenSSH,也有商业实现方案.使用范围最广泛的当然是开源实现OpenSSH. 2. SSH工作原理 在讨论SSH的原理和使用前,我们需要分 ...