题意

给定两个长度为 \(n\) 的序列 \(\{a_i\}\) 与 \(\{b_i\}\),你需要求出它们的相似度。,我们定义这两个序列的相似度为将其中一个序列转化为另一个序列所需的最小操作次数。一次操作定义如下: - 选择序列中的一个长度为 \(k\) 的子段,将子段内的所有元素异或上一个相同的值 \(x\),\(x\) 可以任意决定。特殊地,若其中一个序列无论如何都不能转化到另一个序列,那么这两个序列的相似度为 \(-1\)。 这之后,将会有 \(q\) 次修改操作,单次操作格式如下:

s p v:\(s\) 是一个小写字符,且要么为 a,要么为 b。若 \(s =\) a,则表明将 \(a_p\) 的值修改为 \(v\);若 \(s =\) b,则表明将 \(b_p\) 的值修改为 \(v\)。

在每一次修改操作结束后,你也需要求出两个序列的相似度。输出最开始及每次修改后的答案。

\(1 \leq k \leq n \leq 2 \times 10^5, 0 \leq q \leq 2 \times 10^5, 1 \leq p \leq n, 0 \leq a_i, b_i, v < 2^{14}\),输入保证给出的两个序列长度均为 \(n\)。

分析

  • 先考虑暴力:每次暴力修改,然后贪心地从第一个位置开始推,要修改就修改。

  • 考虑差分,这样每次修改的位置就只有两个了。

  • 将所有位置按照模 \(k\) 分组,每一组互不干扰。我们记某组中连续的三个位置为 1,2,3,\(c\) 数组表示 \(a\) 和 \(b\) 的差分数组的异或结果。每个位置需要异或的值就是 \(c_1; c_1\ xor\ c_2;c_1\ xor\ c_2\ xor c_3\),所以如果有解,每组的异或值都要为0.

  • 讨论 \(k\) 的大小来修改:

    如果 \(k > \sqrt n\) ,每组个数 \(\frac{n}{k}<\sqrt n\) ,暴力即可, 复杂度 \(O(n\sqrt n)\)。

    如果 \(k<\sqrt n\) ,组数 \(k<\sqrt n\) ,每组分块维护,复杂度 \(O(n\sqrt n)\) 。

  • 总时间复杂度为 \(O(n\sqrt n)\)。

代码

#include<bits/stdc++.h>
using namespace std;
typedef long long LL;
#define go(u) for(int i = head[u], v = e[i].to; i; i=e[i].lst, v=e[i].to)
#define rep(i, a, b) for(int i = a; i <= b; ++i)
#define pb push_back
#define re(x) memset(x, 0, sizeof x)
inline int gi() {
int x = 0,f = 1;
char ch = getchar();
while(!isdigit(ch)) { if(ch == '-') f = -1; ch = getchar();}
while(isdigit(ch)) { x = (x << 3) + (x << 1) + ch - 48; ch = getchar();}
return x * f;
}
template <typename T> inline void Max(T &a, T b){if(a < b) a = b;}
template <typename T> inline void Min(T &a, T b){if(a > b) a = b;}
const int N = 2e5 + 7, gg = 400;
int n, k, q, ans, cnt, ndc;
int a[N], b[N], c[N], pre[N], bl[N], L[N], tag[N];
char s[N];
int num[1007][17000];
void upd1(int p, int v) {
for(int i = p; i <= n; i += k) {
pre[i] ^= v;
if(v && pre[i] == 0) --ans;
if(v && pre[i] == v) ++ans;
}
int lst = p % k + ((n - p % k) / k) * k;
if(v && pre[lst] == 0) --cnt;
if(v && pre[lst] == v) ++cnt;
}
void upd2(int p, int v) {
for(int i = p; bl[i] == bl[p]; i += k) {
if(v && (pre[i] ^ tag[bl[p]]) == 0) ++ans;
num[bl[p]][pre[i]] --;
pre[i] ^= v;
num[bl[p]][pre[i]] ++;
if(v && (pre[i] ^ tag[bl[p]]) == 0) --ans;
}
for(int i = bl[p] + 1; i <= ndc && L[i] % k == p % k; ++i) {
if(v) ans += num[i][tag[i]];
tag[i] ^= v;
if(v) ans -= num[i][tag[i]];
}
int lst = p % k + ((n - p % k) / k) * k;
if(v && (pre[lst] ^ tag[bl[lst]]) == 0) --cnt;
if(v && (pre[lst] ^ tag[bl[lst]]) == v) ++cnt;
}
void modify(int p, int v) {
if(n / k < gg)
upd1(p, v ^ c[p]);
else
upd2(p, v ^ c[p]);
c[p] = v;
}
int main() {
n = gi(), k = gi(), q = gi();
rep(i, 1, n) a[i] = gi();
rep(i, 1, n) b[i] = gi();
++n; if(n / k >= gg)
for(int i = 1; i <= k; ++i)
for(int j = i; j <= n; j += k) {
if(((j - 1) / k + 1) % gg == 1) {
L[++ndc] = j;
num[ndc][0] = min(gg, (n - j) / k + 1);
}
bl[j] = ndc;
}
rep(i, 1, n) modify(i, (a[i] ^ a[i - 1]) ^ (b[i] ^ b[i - 1]));
printf("%d\n", cnt ? -1 : ans);
int i, v;
while(q--) {
scanf("%s%d%d", s, &i, &v);
if(s[0] == 'a') a[i] = v;
else b[i] = v;
modify(i, (a[i] ^ a[i - 1]) ^ (b[i] ^ b[i - 1]));
if(i != n) modify(i + 1, (a[i + 1] ^ a[i]) ^ (b[i + 1] ^ b[i]));
printf("%d\n", cnt ? -1 : ans);
}
return 0;
}

[CF1083F]The Fair Nut and Amusing Xor[差分+同余分类+根号分治+分块]的更多相关文章

  1. [cf1083F]The Fair Nut and Amusing Xor

    令$c_{i}=a_{i}\oplus b_{i}$,那么也就是要对$c_{i}$执行操作使其变为0 显然有一个贪心的策略,即从左往右,若当前$c_{i}\ne 0$,则执行对$[i,i+k)$异或$ ...

  2. CF1083E The Fair Nut and Rectangles

    CF1083E The Fair Nut and Rectangles 给定 \(n\) 个平面直角坐标系中左下角为坐标原点,右上角为 \((x_i,\ y_i)\) 的互不包含的矩形,每一个矩形拥有 ...

  3. CF 1083 B. The Fair Nut and Strings

    B. The Fair Nut and Strings 题目链接 题意: 在给定的字符串a和字符串b中找到最多k个字符串,使得不同的前缀字符串的数量最多. 分析:  建出trie树,给定的两个字符串就 ...

  4. CF 1083 A. The Fair Nut and the Best Path

    A. The Fair Nut and the Best Path https://codeforces.com/contest/1083/problem/A 题意: 在一棵树内找一条路径,使得从起点 ...

  5. CF1083A The Fair Nut and the Best Path

    CF1083A The Fair Nut and the Best Path 先把边权搞成点权(其实也可以不用),那么就是询问树上路径的最大权值. 任意时刻权值非负的限制可以不用管,因为若走路径 \( ...

  6. Codeforces Round #526 (Div. 2) E. The Fair Nut and Strings

    E. The Fair Nut and Strings 题目链接:https://codeforces.com/contest/1084/problem/E 题意: 输入n,k,k代表一共有长度为n的 ...

  7. Codeforces Round #526 (Div. 2) D. The Fair Nut and the Best Path

    D. The Fair Nut and the Best Path 题目链接:https://codeforces.com/contest/1084/problem/D 题意: 给出一棵树,走不重复的 ...

  8. Codeforces Round #526 (Div. 2) C. The Fair Nut and String

    C. The Fair Nut and String 题目链接:https://codeforces.com/contest/1084/problem/C 题意: 给出一个字符串,找出都为a的子序列( ...

  9. C. The Fair Nut and String 递推分段形dp

    C. The Fair Nut and String 递推分段形dp 题意 给出一个字符串选择一个序列\({p_1,p_2...p_k}\)使得 对于任意一个\(p_i\) , \(s[p_i]==a ...

随机推荐

  1. Problem4-Project Euler

    Largest palindrome product   A palindromic number reads the same both ways. The largest palindrome m ...

  2. Oracle 与 mysql 建立透明网关

    1.1.1.   文档编写目的 项目上需要做与Mysql数据库的对接,然而Oracle与Mysql数据库是异构的,因此这里采用透明网关的方式来解决. 另,项目上的环境为APP:12.2.6 ; DB: ...

  3. VS2010/2013 运行是很卡的加速方案

    前段时间为了一个项目而把VS2008换成了VS2010,结果原本就不堪重负的本本跑起VS2010来那更是慢得没话说,于是看了遍VS2010选项,又从网上到处找资料找优化方法,总算使我的VS2010跑得 ...

  4. jquery实现显示textarea输入字符数

    起初会想到使用keyup.keydown.keypress或者是onchange事件,onchange需要失去焦点才触发, 其它三个有些对按住键盘某个键不放不生效,有些对使用中文输入法正在输入时统计不 ...

  5. 获取指定时间的Date对象,IE和Chrome的区别(兼容IE)

    网上的大多教程都是 new Date("2016-08-03 00:00:00"); 其实这是Chrome的写法,在IE中并不起作用,在IE中应为 new Date("2 ...

  6. 异常System.BadImageFormatException

    [问题描述] Server Error in '/' Application. Could not load file or assembly 'WebDemo' or one of its depe ...

  7. python容错

    #try: except: else: #为什么叫容错呢,先说说错误,这里说的错误并不是因为马虎或者什么原因在脚本中留下的bug,这个不能容掉,所谓容掉就是略过这个错误,要在测试时候发现并修正,需要容 ...

  8. cd mkdir mv cp rm 命令目录相关操作

    切换目录: cd 家目录 cd. 当前目录 cd.. 当前上一级目录 cd../../当前目录的上上级目录 cd - 返回前一个目录 --------------------------------- ...

  9. 【Weex学习】环境搭建

    教程来源:http://jspang.com/2017/07/12/weex/,我本地是第一次安装Android Studio和教程有些出入 一.软件安装 1.安装Node.js 2.安装Java(h ...

  10. MySQL基础值 存储过程和函数

    一.创建存储过程和函数 什么是创建存储过程和函数? 就是将经常使用的一组SQL语句组合在一起,并将这些SQL语句当做一个整体存储在MYSQL服务器中. 创建存储过程的语句是:CREATE  PROCE ...