「CF1101F」Trucks and Cities

news/2025/1/21 12:57:43/文章来源:https://www.cnblogs.com/keysky/p/18683419

题意描述

\(N\) 座城市,第 \(i\) 座坐标为 \(a_i\) ,有 \(M\) 辆卡车,第 \(i\) 辆卡车要从城市 \(s_i\) 前往城市 \(e_i\) ,每单位长度耗油量为 \(c_i\) ,可以在中途城市加满油 \(r_i\) 次,求让所有卡车都能到达目的地最小的油箱容积。
传送门

思考&做法

step 1

先来思考暴力,二分答案,然后check所有车,时间 \(O(MN\log 10^{18})\) ,会 \(\text{TLE}\)
到这,就有了多个分支,有大佬改换思路用 \(\text{dp}\),用四边形不等式推决策单调性优化 \(\text{dp}\) ,也有大佬用分别二分并随机化加剪枝优化暴力的,这些当时我都没有想到,就瞪大眼睛硬做这道题。

step 2

苦思冥想的时候,yangbaich给了我一个思路:在二分 \(V\) 后首先会预处理 \(f_{i, j}\) 表示在容量为 \(V\) 时,从城市 \(i\) 到城市 \(j\) ,途中不加油,\(c\) 最大是多少,然后暴力是每次加油一次并跳一格,我们要不 \(\text{TLE}\) 就要每次不止跳一格,这时我们或许就会想到倍增,但怎么构建倍增表呢?发现对于卡车 \(i\) ,只会走 \(f_{u, v} \ge c_i\) 的边 \((u, v)\) ,所以考虑对于 \(f\)\(f_{u, v}\) 从大到小排序,卡车按 \(c_i\) 从大到小排序,然后双指针运动,保证在遍历到第 \(i\) 辆卡车时会加入所有该卡车能走的边,并且没有该卡车不能走的边。

step3

本来以为能做了,只不过每次要加边,倍增表只能静态,那怎么做呢?如果没有「CF13E」Holes的经验其实很难想到动态修改的跳表。思路就是按索引分块,对于每个元素维护一个 \(nxt_i\) 表示从 \(i\) 跳一步后的位置,\(jump_i\) 表示从 \(i\) 第一次跳到下一个块的位置,\(step_i\) 表示从 \(i\) 第一次跳到下一个块的过程中跳了多少步,每次修改 \(nxt_i\) 就对所有 \(j \in [L_{bel_i}, i]\) 修改 \(jump_j\)\(jump_{nxt_j} + 1\)\(nxt_j\) ,分讨,并赋值 \(step_j\)
在查询的时候,跳的过程中每次check当前 \(r\) 是否小于 \(step_u\) ,若小,跳出循环,暴力跳 \(r\)\(nxt_i\) ,返回当前点 \(u\) ,然后看 \(u\) 是否大于 \(e_i\) ,若所有卡车都大于,返回 \(\text{true}\) ,反之返回 \(\text{false}\)

概括

二分答案,将卡车和预处理的 \(f\) 表按 \(c_i\)\(f_{i, j}\) 从大到小排序,并用双指针依次加入 \(f_{u, v}\) 的边 \((u, v)\) 并保证 \(\forall f_{u, v} > c_i\) ,用分块维护跳表,让每一辆卡车跳 \(r\) 次,check落点 \(u\) 是否大于 \(e_i\)
总时间复杂度 \(O(M \sqrt{N} \log 10^{18})\)

solution

/*
address:https://codeforces.com/problemset/problem/1101/F
AC 2025/1/21 11:40
*/
#include<bits/stdc++.h>
using namespace std;
typedef long long LL;
const int N = 405, M = 2.5e5 + 5;
const LL INF = 1e18;
int a[N];
int n, m;
struct truck {int s, f, c, r;bool operator < (const truck& o)const { return c > o.c; }
}c[M];
struct node {int x, y;LL c;bool operator < (const node& o)const { return c > o.c; }
}e[N * N];
int siz, blk;
int bel[N], nxt[N], jump[N], step[N], L[N], R[N];
inline void change(int id, int x) {nxt[id] = x;// printf("change %d -> %d ", id, x);for (int i = id;i >= L[bel[id]];i--)if (nxt[i] > R[bel[i]]) step[i] = 1, jump[i] = nxt[i];else step[i] = step[nxt[i]] + 1, jump[i] = jump[nxt[i]];// printf("step:%d\n", step[id]);
}
inline int lift(int id, int x) {while (true) {if (x < step[id]) break;x -= step[id];id = jump[id];// printf("%d ", id);}for (int i = 1;i <= min(x, siz);i++) id = nxt[id];// puts("");return id;
}
inline bool check(LL v) {// printf("%lld:\n", v);int t = 0;for (int i = 1;i <= n;i++)for (int j = i + 1;j <= n;j++)e[++t] = { i, j, v / (a[j] - a[i]) };sort(e + 1, e + t + 1);// for (int i = 1;i <= t;i++) printf("%d %d %lld\n", e[i].x, e[i].y, e[i].c);for (int i = 1;i <= n;i++) nxt[i] = jump[i] = i, step[i] = N;int j = 1;for (int i = 1;i <= m;i++) {// printf("truck %d:", i);while (e[j].c >= c[i].c && j <= t) {change(e[j].x, e[j].y);j++;}// puts("");int u = lift(c[i].s, c[i].r + 1);if (u < c[i].f) return false;}return true;
}
int main() {scanf("%d%d", &n, &m);siz = sqrt(n);blk = (n + siz - 1) / siz;for (int i = 1;i <= n;i++) scanf("%d", &a[i]);for (int i = 1;i <= n;i++) bel[i] = (i + siz - 1) / siz;for (int i = 1;i <= blk;i++) L[i] = (i - 1) * siz + 1, R[i] = min(i * siz, n);for (int i = 1;i <= m;i++) scanf("%d%d%d%d", &c[i].s, &c[i].f, &c[i].c, &c[i].r);sort(c + 1, c + m + 1);LL l = 1, r = INF, v = INF;while (l <= r) {LL mid = l + r >> 1;if (check(mid)) r = mid - 1, v = mid;else l = mid + 1;}printf("%lld", v);return 0;
}

总结

不知道为什么,与yangbaich做题总能做出一些奇奇怪怪的做法,只不过这样也有益于拓展思维,希望后面不知只是按题解所说去做,能多有一些自己的思考并实践。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hqwc.cn/news/872748.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

识别两个表格文件,根据手机号进行匹配相同行并按照需要字段输出

python代码# -*- coding: utf-8 -*- # encoding:utf-8 from flask import Flask, render_template, request, send_file,jsonify import os, requests import pandas as pd from datetime import datetime, timedelta import time, json from log import logging import thread…

语音播报,套件多少异常的问题。(含源代码)

在工作中遇到一家工厂老板的需求:因为产品是有多个配件组成,在生产的时候,经常会多生产,少生产,在组装时,也会出现配件多少的问题,现就此问题设计一款程序。多出,少的,异常的,正常好,会开语音播报。现将全部代码给出以备。 import inspect import os import threadi…

EDMA3学习笔记1

1.1 EDMA3简介 EDMA3(Enhanced Direct Memory Access 3):增强型直接存储器访问的控制器。它是DSP中一个高级数据传输引擎,其结构适合数据的高速传输,可以在没有CPU主要参与的情况下,由控制器完成数据转移,主要服务外部内存(DDR)、片上内存(L2 SRAM)以及串口外设等。…

又一个新项目完结,炸裂!

除了全程直播讲解的、50 个小时的保姆级视频教程之外,我还写了整套文字教程(15 万多字),细致入微!大家好,我是程序员鱼皮。经过了 2 个月的爆肝,我在自己的编程导航的第 11 套有 保姆级教程 的大项目 —— 企业级智能协同云图库平台,完结啦!除了全程直播讲解的、50 个…

kylin V10 SP2 离线单机部署tidb v8.3.0

准备离线组件包 在官方下载页面选择对应版本的 TiDB server 离线镜像包(包含 TiUP 离线组件包)。需要同时下载 TiDB-community-server 软件包和 TiDB-community-toolkit 软件包。 部署离线环境TiUP组件 将离线包传至服务器,执行以下命令安装 TiUP 组件: tar zxvf tidb-comm…

生成型AI应用的质量为何常常不尽人意,以及如何改进

生成型AI应用的质量为何常常不尽人意,以及如何改进2025年,图片来源:elements.envato.com,Marcel Mller 编辑过去两年,生成型AI的热潮席卷了商业世界。这项技术可以提高业务流程的执行效率,减少等待时间,降低过程缺陷。像ChatGPT这样的接口使得与大型语言模型(LLM)的互…

C# WEB API windows server 发布注意事项

1、使用背景: 数据请求方通过接口获取数据,同时使用方通过用户名称进行功能限制;2、实现方法: C# web服务功能,不同机型使用不同接口进行数据获取,请求数据需要包含产品条码信息、请求方用户名信息; 接口请求如下: 3、部署注意事项: 在windows server IIS 管理器中添…

菜单权限的设计与实现

说明该文章是属于OverallAuth2.0系列文章,每周更新一篇该系列文章(从0到1完成系统开发)。该系统文章,我会尽量说的非常详细,做到不管新手、老手都能看懂。说明:OverallAuth2.0 是一个简单、易懂、功能强大的权限+可视化流程管理系统。 友情提醒:本篇文章是属于系列文章,…

第十一章 成本管理(2025年详细解析版)

目录导语章节介绍什么是成本管理?关注两类成本11.1 管理基础项目成本管理的作用和意义项目成本失控原因(了解)成本类型机会成本沉没成本发展趋势和新兴实践如何计算进度偏差?11.2 项目成本管理过程过程概述项目成本管理过程裁剪时需要考虑的因素(不重要)在敏捷或适应型环…

寒假

今天继续学习Android Studio,今天学习的基础空间Button和EditText两个组件,了解了它们的属性,设计了一个简单的登录页面尝试获取前端的输入的信息,目前还未成功,经过学习,觉得融会贯通,目前认为,xml对应javaweb中的前端html页面,activity对应javaweb中的后端中的Contr…