题意

给定两个长度为 \(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. tomcat报错this web application instance has been stopped already问题

    上网搜了下,大部分的报错原因:重启时候 之前的tomcat未正常关闭 ,导致在重启时候 报了这个问题.mac下解决: ps -ef|grep tomcat 找到在进行的tomcat ,kill -9  ...

  2. python常用模块之subprocess

    python常用模块之subprocess python2有个模块commands,执行命令的模块,在python3中已经废弃,使用subprocess模块来替代commands. 介绍一下:comm ...

  3. 更改 Windows VM 的可用性集

    以下步骤说明如何使用 Azure PowerShell 来更改 VM 的可用性集. 只能在创建 VM 时将 VM 添加到可用性集. 如果要更改可用性集,必须将虚拟机删除,并重新创建虚拟机. 使用 Po ...

  4. 关于 Azure Windows VM 的磁盘和 VHD

    就像其他任何计算机一样,Azure 中的虚拟机将磁盘用作存储操作系统.应用程序和数据的位置. 所有 Azure 虚拟机都至少有两个磁盘,即 Windows 操作系统磁盘和临时磁盘. 操作系统磁盘基于映 ...

  5. 使用SQL Server Management Studio操作replication时,要用机器名登录,不要用IP地址

    如果你在使用SSMS(SQL Server Management Studio)登录SQL Server时,使用的是IP地址,如下图所示: 当你操作replication时,会报错: 从上面的错误提示 ...

  6. SQL SERVER中关于OR会导致索引扫描或全表扫描的浅析 (转载)

    在SQL SERVER的查询语句中使用OR是否会导致不走索引查找(Index Seek)或索引失效(堆表走全表扫描 (Table Scan).聚集索引表走聚集索引扫描(Clustered Index ...

  7. tail 尾巴

    tail用法:尾巴,取文件的最后N行,默认前10行, -n 2 取前2行-n 2,简写就是-2 -f 文件 跟踪一个文件尾部的时时变化. 克隆出一个窗口执行:循环脚本:for n in `seq 1 ...

  8. gdb调试:

    1 简介      GDB(GNU Debugger)是GCC的调试工具.其功能强大,现描述如下:      GDB主要帮忙你完成下面四个方面的功能:      1.启动你的程序,可以按照你的自定义的 ...

  9. python3: 数字日期和时间(1)

    ---恢复内容开始--- 1. 数字的四舍五入 Q: 你想对浮点数执行指定精度的舍入运算 A: 简单的使用内置的round(value, ndigits)函数即可. >>> roun ...

  10. EF CodeFirst 数据库的操作

    生成数据库 首先需要通过Nuget安装Migration 这一步是初始化Migration操作. 首先打开程序包控制台, 工具——NuGet包管理器——程序包管理控制台 打开后,在控制台输入下面的命令 ...