前言

DennyQi太巨了!

定义一个点\(a\),\(a_x\)表示\(a\)在第\(x\)维空间上的坐标值

题解

这题的思路珂以说非常巧妙(原谅我又用了这个"珂"),

我们知道曼哈顿距离是\(\sum|a_i-b_i|\),\(|a_i-b_i|\)其实也珂以看作是\((a_i-b_i)\)和\((b_i-a_i)\)中较大的那个。

根据上面的分析曼哈顿距离珂以看作是\(\sum max(a_i-b_i,b_i-a_i)\),

继续分析珂以得出,每个点在每一维度上要应用的无非只有两种情况\(a_i\),\(-a_i\)。

由于区间内取两点求最小值要求每个点都是完整的(就是说一旦选取了该点那么必定会用到该点所有维度上的坐标),那么也就意味着对于一个点,它最多可能(仅为可能)用到的状态会是\(2^k\)次方个。

本题坐标只有5维,电脑不是人脑,那么显然我们珂以枚举解决以上问题。

那么求曼哈顿距离的时候便利所有\(a_i\)的正负性,显然若\(a_i\)为正,\(b_i\)即负;\(a_i\)为负,\(b_i\)为正。

如果用一个二进制状态来压缩,1表示\(a_i\)为正,0表示\(a_i\)为负,显然任意一组关于\(a\)的状态,记为\(sta\),

对应的\(b\)的装态表示为\((((2^k) - 1) - sta)\)也珂以记作\((((2^k) - 1)\ \textbf{xor}\ sta)\)。

那么用一个线段树维护,每一个节点表示对应的区间\([l,r]\)内的曼哈顿距离最大值。

上传标记非常简单就是暴力取两个子节点,然后取对于每一种正负性的珂能,都取max即可。

inline void pushUp(int pos){
for (int i = 0; i < max_sta; ++i)
seg[pos].f[i] = max(seg[pos << 1].f[i], seg[pos << 1 | 1].f[i]);
}

同理应用于合并query时的左右子树的结果。

例子

如果您没有看懂上面的文字,我们不妨来举一个例子。

若\(k=3\)时有两个点\(a,b\)坐标分别记为\((1,2,3),(-1,2,4)\)

那么对于点a来说有\(2^3=8\)种状态,

即\((1,2,3),(1,2,-3),(1,-2,3),(1,-2,-3),\)

  \((-1,2,3),(-1,2,-3),(-1,-2,3),(-1,-2,-3)\)

同理b也有\(2^3=8\)种状态,对于这\(a,b\)的\(8\)种状态两两对应,也就是

\((1,2,3)\)对应\((-(-1),-2,-4)\dots\)这些互相对应的状态取max以后也就是 \(\sum|a_i-b_i|\)啦QAQ

代码

#include <cstdio>
#include <cstring>
#include <algorithm> using namespace std; struct Pos{
int f[32];
} seg[800005]; int n, k, max_sta; inline void pushUp(int pos){
for (int i = 0; i < max_sta; ++i)
seg[pos].f[i] = max(seg[pos << 1].f[i], seg[pos << 1 | 1].f[i]);
} int tmppos[5]; void build(int pos, int l, int r){
if (l == r){
for (int i = 0; i < k; ++i) scanf("%d", &tmppos[i]);
for (int i = 0; i < max_sta; ++i){
seg[pos].f[i] = 0;
for (int j = 0; j < k; ++j)
if (i & (1 << j))
seg[pos].f[i] += tmppos[j];
else
seg[pos].f[i] -= tmppos[j];
}
return ;
}
int mid = (l + r) >> 1;
build(pos << 1, l, mid), build(pos << 1 | 1, mid + 1, r);
pushUp(pos);
} inline void mrg(Pos &a, Pos b){
for (int i = 0; i < max_sta; ++i)
a.f[i] = max(a.f[i], b.f[i]);
} Pos query(int pos, int l, int r, int x, int y){
if (x <= l && r <= y) return seg[pos];
int mid = (l + r) >> 1;
Pos res;
if (x <= mid)
res = query(pos << 1, l, mid, x, y);
if (y > mid){
if (x <= mid)
mrg(res, query(pos << 1 | 1, mid + 1, r, x, y));
else
res = query(pos << 1 | 1, mid + 1, r, x, y);
}
return res;
} void modify(int pos, int l, int r, int x){
if (l == r){
for (int i = 0; i < max_sta; ++i){
seg[pos].f[i] = 0;
for (int j = 0; j < k; ++j)
if (i & (1 << j))
seg[pos].f[i] += tmppos[j];
else
seg[pos].f[i] -= tmppos[j];
}
return ;
}
int mid = (l + r) >> 1;
if (x <= mid) modify(pos << 1, l, mid, x);
else modify(pos << 1 | 1, mid + 1, r, x);
pushUp(pos);
} int main(){
scanf("%d %d", &n, &k); max_sta = 1 << k;
build(1, 1, n);
int q; scanf("%d", &q);
while (q--){
int op; scanf("%d", &op);
int i;
if (op == 1){
scanf("%d", &i);
for (int j = 0; j < k; ++j)
scanf("%d", &tmppos[j]);
modify(1, 1, n, i);
}
else if (op == 2){
int l, r; scanf("%d %d", &l, &r);
Pos res = query(1, 1, n, l, r);
int ans = 0;
for (int i = 0; i < (1 << (k - 1)); ++i)
ans = max(ans, res.f[i] + res.f[(max_sta - 1) ^ i]);
printf("%d\n", ans);
}
}
return 0;
}

[CF1093G]Multidimensional Queries 题解的更多相关文章

  1. [CF1093G]Multidimensional Queries

    [CF1093G]Multidimensional Queries 题目大意: \(k(k\le5)\)维空间中有\(n(n\le2\times10^5)\)个点.\(m\)次操作,操作包含一下两种: ...

  2. [CF1093G]Multidimensional Queries:线段树

    分析 非常有趣的一道题. 式子中的绝对值很难处理,但是我们发现: \[\sum_{i=1}^{k}|a_{x,i}-a_{y,i}|=\sum_{i=1}^{k}max(a_{x,i}-a_{y,i} ...

  3. CF 1093 G. Multidimensional Queries

    G. Multidimensional Queries 链接 分析: 考虑如何去掉绝对值符号. $\sum \limits_{i = 1}^{k} |a_{x, i} - a_{y, i}|$,由于k ...

  4. CF 1093G Multidimensional Queries——线段树(消去绝对值符号)

    题目:http://codeforces.com/contest/1093/problem/G 只好看看题解:https://codeforces.com/blog/entry/63877 主要是把绝 ...

  5. CF375D Tree and Queries 题解

    感觉CF的题目名都好朴素的样子 你谷链接 首先这题显然是个dsu on tree 但是我不会. 其次这题显然是个莫队.这我会啊! 然后会发现好像不是很对劲.因为每次询问都有一个k,貌似和传统的莫队数颜 ...

  6. POJ-2926-Requirements&&Educational Codeforces Round 56G. Multidimensional Queries 【哈夫曼距离】

    POJ2926 先学会这个哈夫曼距离的处理才能做 cf 的G #include <iostream> #include <stdio.h> #include <algor ...

  7. codeforces1093G Multidimensional Queries 【线段树】

    题目分析: 搜索2^k种情况,线段树分别处理就行了,正确性明显. 代码: #include<bits/stdc++.h> using namespace std; ; int n,k; ] ...

  8. CodeForces - 1093G:Multidimensional Queries (线段树求K维最远点距离)

    题意:给定N个K维的点,Q次操作,或者修改点的坐标:或者问[L,R]这些点中最远的点. 思路:因为最后一定可以表示维+/-(x1-x2)+/-(y1-y2)+/-(z1-z2)..... 所以我们可以 ...

  9. Multidimensional Queries(二进制枚举+线段树+Educational Codeforces Round 56 (Rated for Div. 2))

    题目链接: https://codeforces.com/contest/1093/problem/G 题目: 题意: 在k维空间中有n个点,每次给你两种操作,一种是将某一个点的坐标改为另一个坐标,一 ...

随机推荐

  1. Excel透视表基础之数据源、创建、基本术语、基本操作

    数据源的基本要求: 每列数据的第一行包含该列标题 不能包含空行或空列 不能包含空单元格 不能包含合并单元格 不能包含同类字段 如果包含空行.空列则删除空行和空列.如果包含空单元格则填充空单元格. 如果 ...

  2. adb 打印kernel输出的log

     一. linux 内核printk机制     1.1. Android内核是基于Linxu kernel的,因此其log机制也是通用的,在Android内核中使用printk函数进行Log输出.与 ...

  3. redis 命令都在这了

    DEL key [key ...]删除指定的key(一个或多个) DUMP key导出key的值 EXISTS key [key ...]查询一个key是否存在 EXPIRE key seconds设 ...

  4. [LeetCode] 227. 基本计算器 II

    题目链接: https://leetcode-cn.com/problems/basic-calculator-ii 难度:中等 通过率:33.2% 题目描述: 实现一个基本的计算器来计算一个简单的字 ...

  5. Genymotion模拟器使用camera

    1.前言 最近开发react-native的app,上传图片功能需要使用相机,发现Genymotion默认的相机不工作.查看同行的博客解决了,归纳整理一下. 2.步骤 2.1安装Genymotion: ...

  6. Java日志打印方法

    一.使用log4j打印日志 1. 下载log4j.jar和commons-logging.jar.     log4j.jar下载地址:http://logging.apache.org/log4j/ ...

  7. lua基础(2)

    错误处理: local function add(a,b) assert(type(a) == "number", "a 不是一个数字") assert(typ ...

  8. Arm宣布将在Linux中应用事务内存扩展支持

    事务内存扩展是Arm对硬件事务内存的支持,用于改进对大型共享数据结构的并发访问. 在宣布事务性内存扩展和可伸缩向量扩展2时,Arm表示他们将很快为GNU工具链和LLVM提供支持.正如我们已经看到的,为 ...

  9. Qualcomm_Mobile_OpenCL.pdf 翻译-6-工作组尺寸的性能优化

    对于许多kernels来说,工作组大小的调整会是一种简单有效的方法.这章将会介绍基于工作组大小的基础知识,比如如何获取工作组大小,为什么工作组大小非常重要,同时也会讨论关于最优工作组大小的选择和调整的 ...

  10. Rsync实现负载均衡的数据同步

    使用三台服务器:系统:CentOS 6.8 192.168.8.169 开发服务器 192.168.8.167 线上服务器1192.168.8.168 线上服务器2 实现思路:在开发服务器上制定一个规 ...