D-Distance_2019牛客暑期多校训练营(第八场)
题目链接
题意
1<=nmh,q<=1e5
q个操作
- 1 x y z往坐标里加入一个点
- 2 x y z查询距离该点最近的点的距离(曼哈顿距离)
题解
做法一
将要插入的点保存在一个队列中,当队列中的点数达到一个阈值就把队列中所有点取出,暴力的bfs一次把答案记录在\(dis[getid(x,y,z)]\)中表示距离点\((x,y,z)\)最近的点的距离,查询的时候就取暴力的计算查询点和队列中每个点的距离,再和已经插入的点也就是dis数组取最小值,当阈值取\(\sqrt{nmh}\)时复杂度为\(O(\frac{qnmh}E + qE) = O(nmh + q\sqrt{nmh})\)
做法二
将距离公式\(|x_0-x_i| + |y_0-y_i| + |z_0-z_i|\)的绝对值拆开有八种情况\(\pm(x_0-x_i) \pm (y_0-y_i) \pm (z_0-z_i)\)
这八种情况的最大值就是真正的距离,我们将插入的点分成\((\pm x, \pm y, \pm z)\)八种情况分别插入八个树状数组,树状数组维护\(x_i <= x, y_i <= y, z_i <= z的x+y+z\)的最大值,目的是为了把求最近点的距离转换成求\((x+y+z-x_i-y_i-z_i)\)的最小值,查询的时候对八个树状数组答案取min就行了
代码
做法一
#include <bits/stdc++.h>
using namespace std;
typedef long long ll;
const int mx = 1e5+5;
const int mod = 998244353;
const int INF = 0x3f3f3f3f;
const int dir[6][3] = {{0,0,1}, {0,0,-1}, {0,1,0}, {0,-1,0}, {1,0,0}, {-1,0,0}};
int dis[mx];
int n, m, h, q;
struct node {
    int x, y, z, step;
};
vector <node> v;
queue <node> Q;
int getid(int x, int y, int z) {
    return (x-1)*m*h + (y-1)*h + z;
}
int getdis(int x, int y, int z, int a, int b, int c) {
    return abs(x-a) + abs(y-b) + abs(z-c);
}
int main() {
    for (int i = 0; i < mx; i++) dis[i] = INF;
    scanf("%d%d%d%d", &n, &m, &h, &q);
    int sq = sqrt(n*m*h)    ;
    while (q--) {
        int op, x, y, z;
        scanf("%d%d%d%d", &op, &x, &y, &z);
        if (op == 1) {
            v.push_back({x, y, z, 0});
            if (v.size() == sq) {
                for (int i = 0; i < v.size(); i++) {
                    Q.push(v[i]);
                    dis[getid(v[i].x, v[i].y, v[i].z)] = 0;
                }
                v.clear();
                while (!Q.empty()) {
                    node now = Q.front();
                    node next;
                    Q.pop();
                    for (int i = 0; i < 6; i++) {
                        next.x = now.x + dir[i][0];
                        next.y = now.y + dir[i][1];
                        next.z = now.z + dir[i][2];
                        if (next.x < 1 || next.x > n || next.y < 1 || next.y > m || next.z < 1 || next.z > h) continue;
                        next.step = now.step + 1;
                        if (next.step < dis[getid(next.x, next.y, next.z)]) {
                            dis[getid(next.x, next.y, next.z)] = next.step;
                            Q.push(next);
                        }
                    }
                }
            }
        } else {
            int ans = dis[getid(x, y, z)];
            for (int i = 0; i < v.size(); i++) {
                ans = min(ans, getdis(x, y, z, v[i].x, v[i].y, v[i].z));
            }
            printf("%d\n", ans);
        }
    }
    return 0;
}
做法二
#include <bits/stdc++.h>
using namespace std;
typedef long long ll;
const int mx = 8e5+5;
const int mod = 998244353;
const int INF = 0x3f3f3f3f;
const int dir[8][3] = {{1,1,1}, {1,1,-1}, {1,-1,1}, {1,-1,-1}, {-1,1,1}, {-1,1,-1}, {-1,-1,1}, {-1,-1,-1}};
int dis[mx];
int n, m, h, q;
int getid(int x, int y, int z) {
    return (x-1)*m*h + (y-1)*h + z;
}
struct Bit {
    int a[mx];
    int lowbit(int x) {
        return x & -x;
    }
    void update(int x, int y, int z) {
        if (x < 0) x += n;
        if (y < 0) y += m;
        if (z < 0) z += h;
        for (int i = x; i <= n; i+=lowbit(i))
            for (int j = y; j <= m; j+=lowbit(j))
                for (int k = z; k <= h; k+=lowbit(k))
                    a[getid(i,j,k)] = max(a[getid(i,j,k)], x+y+z);
    }
    int query(int x, int y, int z) {
        if (x < 0) x += n;
        if (y < 0) y += m;
        if (z < 0) z += h;
        int ans = 0;
        for (int i = x; i > 0; i-=lowbit(i))
            for (int j = y; j > 0; j-=lowbit(j))
                for (int k = z; k > 0; k-=lowbit(k))
                    ans = max(a[getid(i,j,k)], ans);
        if (ans == 0) return INF;
        else return x+y+z-ans;
    }
}bit[8];
int main() {
    scanf("%d%d%d%d", &n, &m, &h, &q);
    n++; m++; h++;
    while (q--) {
        int op, x, y, z;
        scanf("%d%d%d%d", &op, &x, &y, &z);
        if (op == 1) {
            for (int i = 0; i < 8; i++) {
                bit[i].update(x*dir[i][0], y*dir[i][1], z*dir[i][2]);
            }
        } else {
            int ans = INF;
            for (int i = 0; i < 8; i++) ans = min(ans, bit[i].query(x*dir[i][0], y*dir[i][1], z*dir[i][2]));
            printf("%d\n", ans);
        }
    }
    return 0;
}
D-Distance_2019牛客暑期多校训练营(第八场)的更多相关文章
- 2019牛客暑期多校训练营(第九场)A:Power of Fibonacci(斐波拉契幂次和)
		题意:求Σfi^m%p. zoj上p是1e9+7,牛客是1e9: 对于这两个,分别有不同的做法. 前者利用公式,公式里面有sqrt(5),我们只需要二次剩余求即可. 后者mod=1e9,5才 ... 
- 2019牛客暑期多校训练营(第一场)A题【单调栈】(补题)
		链接:https://ac.nowcoder.com/acm/contest/881/A来源:牛客网 题目描述 Two arrays u and v each with m distinct elem ... 
- 2019牛客暑期多校训练营(第一场) B	Integration (数学)
		链接:https://ac.nowcoder.com/acm/contest/881/B 来源:牛客网 Integration 时间限制:C/C++ 2秒,其他语言4秒 空间限制:C/C++ 5242 ... 
- 2019牛客暑期多校训练营(第一场) A	Equivalent Prefixes ( st 表 + 二分+分治)
		链接:https://ac.nowcoder.com/acm/contest/881/A 来源:牛客网 Equivalent Prefixes 时间限制:C/C++ 2秒,其他语言4秒 空间限制:C/ ... 
- 2019牛客暑期多校训练营(第二场)F.Partition problem
		链接:https://ac.nowcoder.com/acm/contest/882/F来源:牛客网 Given 2N people, you need to assign each of them ... 
- 2019牛客暑期多校训练营(第一场)A	Equivalent Prefixes(单调栈/二分+分治)
		链接:https://ac.nowcoder.com/acm/contest/881/A来源:牛客网 Two arrays u and v each with m distinct elements ... 
- [状态压缩,折半搜索] 2019牛客暑期多校训练营(第九场)Knapsack Cryptosystem
		链接:https://ac.nowcoder.com/acm/contest/889/D来源:牛客网 时间限制:C/C++ 2秒,其他语言4秒 空间限制:C/C++ 262144K,其他语言52428 ... 
- 2019牛客暑期多校训练营(第二场)J-Subarray(思维)
		>传送门< 前言 这题我前前后后看了三遍,每次都是把网上相关的博客和通过代码认真看了再思考,然并卵,最后终于第三遍也就是现在终于看懂了,其实懂了之后发现其实没有那么难,但是的的确确需要思维 ... 
- J-Subarray_2019牛客暑期多校训练营(第二场)
		题意 有一个只由1,-1组成的数组,给出所有连续的1所在位置,求满足1的个数大于-1的个数的子区间的数量 题解 参考博客:https://www.cnblogs.com/Yinku/p/1122149 ... 
- 2019牛客暑期多校训练营(第一场)-A (单调栈)
		题目链接:https://ac.nowcoder.com/acm/contest/881/A 题意:给定两个长度均为n的数组a和b,求最大的p使得(a1,ap)和(b1,bp)等价,等价的定义为其任意 ... 
随机推荐
- Appium自动化测试环境搭建
			前言 Appium是一个开源的自动化测试框架,支持跨平台,支持多种编程语言,可用于原生,混合和移动web应用程序,使用webdriver驱动ios,android应用程序.那么为了学习app自动化测试 ... 
- 隐马尔科夫模型HMM介绍
			马尔科夫链是描述状态转换的随机过程,该过程具备“无记忆”的性质:即当前时刻$t$的状态$s_t$的概率分布只由前一时刻$t-1$的状态$s_{t-1}$决定,与时间序列中$t-1$时刻之前的状态无关. ... 
- Windows 下配置 Vagrant 环境
			Vagrant是一个基于 Ruby 的工具,用于创建和部署虚拟化开发环境.它使用 Oracle 的开源VirtualBox虚拟化系统. Vagrant 在快速搭建开发环境方面是很赞的,试想一个团队中, ... 
- selenium操作cookies实现免密登录,自动发微博
			一直想用selenium实现个小功能,比如发微博之类的,但是有的网站在登录会有验证码,没想到太好的方法解决,于是想到利用cookies来登录网站 第一步:获取一个可用的cookies,获取的cooki ... 
- git bash 初始化配置
			这里只针对 windows 下,使用git 时的一些初始配置 1. git bash 安装 下载地址: https://git-for-windows.github.io/ 根据提示,一步步安装即可 ... 
- JS 中构造函数和普通函数的区别
			原来只是随意的了解了下 , 但是最近有点忘了 于是详细了解下 加深下印象. 1.构造函数也是一个普通函数,创建方式和普通函数一样,但构造函数习惯上首字母大写 2.构造函数和普通函数的区别在于:调用方式 ... 
- vue之手把手教你写日历组件
			---恢复内容开始--- 1.日历组件 1.分析功能:日历基本功能,点击事件改变日期,样式的改变 1.结构分析:html 1.分为上下两个部分 2.上面分为左按钮,中间内容展示,右按钮 下面分为周几展 ... 
- Element UI系列:Select下拉框实现默认选择
			实现的主要关键点在于 v-mode 所绑定的值,必须是 options 数组中对应的 value 值 
- 使用IDEA打包scala程序并在spark中运行
			一.首先配置ssh无秘钥登陆, 先使用这条命令:ssh-keygen,然后敲三下回车: 然后使用cd .ssh进入 .ssh这个隐藏文件夹: 再创建一个文件夹authorized_keys,使用命令t ... 
- Redis——发布和订阅
			发布与订阅(又称pub/sub),订阅者(listener)负责订阅频道(channel),发送者(publisher)负责向频道发送二进制字符串消息(binary string message).每 ... 
